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


Posted in PHP onFebruary 23, 2020

本文实例讲述了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服务操作示例

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

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

希望本文所述对大家PHP程序设计有所帮助。

PHP 相关文章推荐
PHP 已经成熟
Dec 04 PHP
PHP数组 为文章加关键字连接 文章内容自动加链接
Dec 29 PHP
mysqli_set_charset和SET NAMES使用抉择及优劣分析
Jan 13 PHP
php curl选项列表(超详细)
Jul 01 PHP
PHP中isset()和unset()函数的用法小结
Mar 11 PHP
zf框架的校验器使用使用示例(自定义校验器和校验器链)
Mar 13 PHP
PHPThumb图片处理实例
May 03 PHP
PHP+MySQL修改记录的方法
Jan 21 PHP
php写入数据到CSV文件的方法
Mar 14 PHP
php调用自己java程序的方法详解
May 13 PHP
php事件驱动化设计详解
Nov 10 PHP
Laravel 解决composer相关操作提示php相关异常的问题
Oct 23 PHP
php 使用ActiveMQ发送消息,与处理消息操作示例
Feb 23 #PHP
php实现通过stomp协议连接ActiveMQ操作示例
Feb 23 #PHP
php ActiveMQ的安装与使用方法图文教程
Feb 23 #PHP
php 多进程编程父进程的阻塞与非阻塞实例分析
Feb 22 #PHP
php7 图形用户界面GUI 开发示例
Feb 22 #PHP
PHP Beanstalkd消息队列的安装与使用方法实例详解
Feb 21 #PHP
PHP pthreads v3在centos7平台下的安装与配置操作方法
Feb 21 #PHP
You might like
咖啡是不是喝了会上瘾?咖啡是必须品吗!
2021/03/04 新手入门
php反弹shell实现代码
2009/04/22 PHP
PHPEXCEL 使用小记
2013/01/06 PHP
php的4种常用运行方式详解
2016/12/22 PHP
XML的代替者----JSON
2007/07/21 Javascript
解决Jquery鼠标经过不停滑动的问题
2014/03/03 Javascript
Javscript删除数组中指定元素并返回新数组
2014/03/06 Javascript
JavaScript和CSS交互的方法汇总
2014/12/02 Javascript
AngularJS转换响应内容
2016/01/27 Javascript
jQuery+ajax简单实现文件上传的方法
2016/06/03 Javascript
vue插件tab选项卡使用小结
2016/10/27 Javascript
JavaScript实现的选择排序算法实例分析
2017/04/14 Javascript
Vue computed计算属性的使用方法
2017/07/14 Javascript
JavaScript寄生组合式继承原理与用法分析
2019/01/11 Javascript
JS设置自定义快捷键并实现图片上下左右移动
2019/10/17 Javascript
react实现移动端下拉菜单的示例代码
2020/01/16 Javascript
浅谈webpack构建工具配置和常用插件总结
2020/05/11 Javascript
关于element-ui表单中限制输入纯数字的解决方式
2020/09/08 Javascript
[00:50]2014DOTA2国际邀请赛 NEWBEE战队回顾
2014/08/01 DOTA
python用ConfigObj读写配置文件的实现代码
2013/03/04 Python
python 读写文件,按行修改文件的方法
2018/07/12 Python
Python实现微信翻译机器人的方法
2019/08/13 Python
python文件操作的简单方法总结
2019/11/07 Python
Python+appium框架原生代码实现App自动化测试详解
2020/03/06 Python
Python configparser模块操作代码实例
2020/06/08 Python
Python如何在单元测试中给对象打补丁
2020/08/03 Python
python 发送邮件的四种方法汇总
2020/12/02 Python
Calzedonia美国官网:意大利风格袜子、打底裤和沙滩装
2018/07/19 全球购物
新西兰最大、占有率最高的综合性药房:PharmacyDirect药房中文网
2020/11/03 全球购物
团队精神的演讲稿
2014/05/14 职场文书
文化苦旅读书笔记
2015/06/29 职场文书
教你快速开启Apache SkyWalking的自监控
2021/04/25 Servers
解决Tkinter中button按钮未按却主动执行command函数的问题
2021/05/23 Python
HTML+VUE分页实现炫酷物联网大屏功能
2021/05/27 Vue.js
golang使用map实现去除重复数组
2022/04/14 Golang
python pandas 解析(读取、写入)CSV 文件的操作方法
2022/12/24 Python