欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

PHP实现创建一个RPC服务操作示例

程序员文章站 2022-05-25 23:01:45
本文实例讲述了php实现创建一个rpc服务操作。分享给大家供大家参考,具体如下:rpc全称为remote procedure call,翻译过来为"远程过程调用"。主要应用于不同的系统之间的远程通信和...

本文实例讲述了php实现创建一个rpc服务操作。分享给大家供大家参考,具体如下:

rpc全称为remote procedure call,翻译过来为"远程过程调用"。主要应用于不同的系统之间的远程通信和相互调用。

比如有两个系统,一个是php写的,一个是java写的,而php想要调用java中的某个类的某个方法,这时候就需要用到rpc了。

怎么调?直接调是不可能,只能是php通过某种自定义协议请求java的服务,java解析该协议,在本地实例化类并调用方法,然后把结果返回给php。

这里我们用php的socket扩展来创建一个服务端和客户端,演示调用过程。

rpcserver.php代码如下:

<?php
class rpcserver {
  protected $serv = null;

  public function __construct($host, $port, $path) {
    //创建一个tcp socket服务
    $this->serv = stream_socket_server("tcp://{$host}:{$port}", $errno, $errstr);
    if (!$this->serv) {
      exit("{$errno} : {$errstr} \n");
    }
    //判断我们的rpc服务目录是否存在
    $realpath = realpath(__dir__ . $path);
    if ($realpath === false || !file_exists($realpath)) {
      exit("{$path} error \n");
    }

    while (true) {
      $client = stream_socket_accept($this->serv);

      if ($client) {
        //这里为了简单,我们一次性读取
        $buf = fread($client, 2048);
        //解析客户端发送过来的协议
        $classret = preg_match('/rpc-class:\s(.*);\r\n/i', $buf, $class);
        $methodret = preg_match('/rpc-method:\s(.*);\r\n/i', $buf, $method);
        $paramsret = preg_match('/rpc-params:\s(.*);\r\n/i', $buf, $params);
        
        if($classret && $methodret) {
          $class = ucfirst($class[1]);
          $file = $realpath . '/' . $class . '.php';
          //判断文件是否存在,如果有,则引入文件
          if(file_exists($file)) {
            require_once $file;
            //实例化类,并调用客户端指定的方法
            $obj = new $class();
            //如果有参数,则传入指定参数
            if(!$paramsret) {
              $data = $obj->$method[1]();
            } else {
              $data = $obj->$method[1](json_decode($params[1], true));
            }
            //把运行后的结果返回给客户端
            fwrite($client, $data);
          }
        } else {
          fwrite($client, 'class or method error');
        }
        //关闭客户端
        fclose($client);
      }
    }
  }

  public function __destruct() {
    fclose($this->serv);
  }
}

new rpcserver('127.0.0.1', 8888, './service');

rpcclient.php代码如下:

<?php

class rpcclient {
  protected $urlinfo = array();
  
  public function __construct($url) {
    //解析url
    $this->urlinfo = parse_url($url);
    if(!$this->urlinfo) {
      exit("{$url} error \n");
    }
  }
  
  public function __call($method, $params) {
    //创建一个客户端
    $client = stream_socket_client("tcp://{$this->urlinfo['host']}:{$this->urlinfo['port']}", $errno, $errstr);
    if (!$client) {
      exit("{$errno} : {$errstr} \n");
    }
    //传递调用的类名
    $class = basename($this->urlinfo['path']);
    $proto = "rpc-class: {$class};" . php_eol;
    //传递调用的方法名
    $proto .= "rpc-method: {$method};" . php_eol;
    //传递方法的参数
    $params = json_encode($params);
    $proto .= "rpc-params: {$params};" . php_eol;
    //向服务端发送我们自定义的协议数据
    fwrite($client, $proto);
    //读取服务端传来的数据
    $data = fread($client, 2048);
    //关闭客户端
    fclose($client);
    return $data;
  }
}

$cli = new rpcclient('http://127.0.0.1:8888/test');
echo $cli->hehe();
echo $cli->hehe2(array('name' => 'test', 'age' => 27));

然后分别运行上面两个脚本(注意,php要添加环境变量)

> php rpcserver.php
> php rpcclient.php

结果如下:

PHP实现创建一个RPC服务操作示例

PHP实现创建一个RPC服务操作示例

test.php代码如下:

<?php
class test {
  public function hehe() {
    return 'hehe';
  }
  public function hehe2($params) {
    return json_encode($params);
  }
}

目录结构如下:

PHP实现创建一个RPC服务操作示例

上面我们自定义的协议,可以随意修改,只要是客户端和服务端两边能够统一并能解析。

客户端通过请求服务端,把要调用的类,方法和参数传递给服务端,服务端去通过实例化调用方法返回结果。