Php怎么使用rpc,使用PHP来简单的创建一个RPC服务

RPC全称为Remote Procedure Call,翻译过来为"远程过程调用"。主要应用于不同的系统之间的远程通信和相互调用。

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

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

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

RpcServer.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}");

}

//判断我们的RPC服务目录是否存在 $realPath = realpath(__DIR__ . $path);

if ($realPath === false || !file_exists($realPath)) {

exit("{$path}error");

}

while (true) {

$client = stream_socket_accept($this->serv);

if ($client) {

//这里为了简单,我们一次性读取 $buf = fread($client, 2048);

//解析客户端发送过来的协议 $classRet = preg_match("/Rpc-Class:s(.*);

/i", $buf, $class);

$methodRet = preg_match("/Rpc-Method:s(.*);

/i", $buf, $method);

$paramsRet = preg_match("/Rpc-Params:s(.*);

/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代码如下:

class RpcClient {

protected $urlInfo = array();

public function __construct($url) {

//解析URL $this->urlInfo = parse_url($url);

if(!$this->urlInfo) {

exit("{$url}error");

}

}

public function __call($method, $params) {

//创建一个客户端 $client = stream_socket_client("tcp://{$this->urlInfo["host"]}:{$this->urlInfo["port"]}", $errno, $errstr);

if (!$client) {

exit("{$errno}:{$errstr}");

}

//传递调用的类名 $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

结果如下:

d07cbf238be30f66e6f4974a37982401.png

a60afec0e319cb343c284693d2c09630.png

Test.php代码如下:

class Test {

public function hehe() {

return "hehe";

}

public function hehe2($params) {

return json_encode($params);

}

}

目录结构如下:

afb33768ae9afc1563ff26e87faec0ca.png

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

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

以上内容就到这里结束, 以上内容希望帮助到大家,很多PHPer在进阶的时候总会遇到一些问题和瓶颈,业务代码写多了没有方向感,不知道该从那里入手去提升,对此我整理了一些资料,包括但不限于:分布式架构、高可扩展、高性能、高并发、服务器性能调优、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql优化、shell脚本、Docker、微服务、Nginx等多个知识点高级进阶干货需要的可以免费分享给大家,需要戳这里

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值