PHP创建简单RPC服务

RPC(Remote Procedure Call)即远程过程调用,指被调用方法的具体实现不在程序运行本地,而是在别的某个地方。主要应用于不同的系统之间的远程通信和相互调用。

如 A 调用 B 提供的 remoteAdd 方法:

首先A与B之间建立一个TCP连接;

然后A把需要调用的方法名(这里是remoteAdd)以及方法参数(10, 20)序列化成字节流发送出去;

B接受A发送过来的字节流,然后反序列化得到目标方法名,方法参数,接着执行相应的方法调用(可能是localAdd)并把结果30返回;

A接受远程调用结果

有些远程调用选择比较底层的 socket 协议,有些远程调用选择比较上层的 HTTP 协议。

远程调用的好处:

解耦:当方法提供者需要对方法内实现修改时,调用者完全感知不到,不用做任何变更;这种方式在跨部门,跨公司合作的时候经常用到,并且方法的提供者我们通常称为:服务的暴露方

这里使用 PHP Socket 来创建一个服务端和客户端,目录结构如下:

PHP创建简单RPC服务

服务端

<?php class RpcServer { protected $server = null; public function __construct($host, $port, $path) { // 创建一个 Socket 服务 if(($this->server = socket_create(AF_INET,SOCK_STREAM,SOL_TCP)) < 0) { exit("socket_create() 失败的原因是:".socket_strerror($this->server)."\n"); } if(($ret = socket_bind($this->server,$host,$port)) < 0) { exit("socket_bind() 失败的原因是:".socket_strerror($ret)."\n"); } if(($ret = socket_listen($this->server,3)) < 0) { exit("socket_listen() 失败的原因是:".socket_strerror($ret)."\n"); } // 判断 RPC 服务目录是否存在 $realPath = realpath(__DIR__ . $path); if ($realPath === false || !file_exists($realPath)) { exit("{$path} error \n"); } do { $client = socket_accept($this->server); if($client) { // 一次性读取 $buf = socket_read($client, 8024); echo $buf; //解析客户端发送过来的协议 $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]); $method = $method[1]; $params = json_decode($params[1], true); $file = $realPath . \'/\' . $class . \'.php\'; // 类文件需要和类名一致 $data = \'\'; // 执行结果 // 判断类文件是否存在 if(file_exists($file)) { // 引入类文件 require_once $file; // 实例化类 $rfc_obj = new ReflectionClass($class); // 判断该类指定方法是否存在 if($rfc_obj->hasMethod($method)) { // 执行类方法 $rfc_method = $rfc_obj->getMethod($method); $data = $rfc_method->invokeArgs($rfc_obj->newInstance(), [$params]); } else { socket_write($client, \'method error\'); } //把运行后的结果返回给客户端 socket_write($client, $data); } } else { socket_write($client, \'class or method error\'); } // 关闭客户端 socket_close($client); } }while(true); } public function __destruct() { socket_close($this->server); } } new RpcServer(\'127.0.0.1\',8080,\'./service\');

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/zzdwyp.html