Home  >  Article  >  Backend Development  >  Analyze how PHP can quickly create RPC services (code demonstration)

Analyze how PHP can quickly create RPC services (code demonstration)

藏色散人
藏色散人forward
2022-11-09 16:18:143158browse

This article will introduce to you what RPC is and how to use PHP to create an RPC service simply and quickly. It is actually very simple. Let’s take a look~ I hope it will be helpful to friends in need~

Analyze how PHP can quickly create RPC services (code demonstration)

RPC stands for Remote Procedure Call, which translates to "remote procedure call". Mainly used for remote communication and mutual calls between different systems. [Recommended learning: PHP video tutorial]

For example, there are two systems, one is written in PHP and the other is written in JAVA, and PHP wants to call a certain class in JAVA For a certain method, RPC needs to be used at this time.

How to adjust? Direct adjustment is impossible. PHP can only request JAVA's services through some custom protocol. JAVA parses the protocol, instantiates the class locally and calls the method, and then returns the result to PHP.

Here we use PHP's socket extension to create a server and client to demonstrate the calling process.

RpcServer.php code is as follows:

<?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(&#39;/Rpc-Class:\s(.*);\r\n/i&#39;, $buf, $class);
                $methodRet = preg_match(&#39;/Rpc-Method:\s(.*);\r\n/i&#39;, $buf, $method);
                $paramsRet = preg_match(&#39;/Rpc-Params:\s(.*);\r\n/i&#39;, $buf, $params);
                 
                if($classRet && $methodRet) {
                    $class = ucfirst($class[1]);
                    $file = $realPath . &#39;/&#39; . $class . &#39;.php&#39;;
                    //判断文件是否存在,如果有,则引入文件
                    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, &#39;class or method error&#39;);
                }
                //关闭客户端
                fclose($client);
            }
        }
    }
 
    public function __destruct() {
        fclose($this->serv);
    }
}
 
new RpcServer(&#39;127.0.0.1&#39;, 8888, &#39;./service&#39;);

RpcClient.php code is as follows:

<?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[&#39;host&#39;]}:{$this->urlInfo[&#39;port&#39;]}", $errno, $errstr);
        if (!$client) {
            exit("{$errno} : {$errstr} \n");
        }
        //传递调用的类名
        $class = basename($this->urlInfo[&#39;path&#39;]);
        $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(&#39;http://127.0.0.1:8888/test&#39;);
echo $cli->hehe();
echo $cli->hehe2(array(&#39;name&#39; => &#39;test&#39;, &#39;age&#39; => 27));

Then run the above two scripts respectively (note that php needs to add environment variables)

> php RpcServer.php
> php RpcClient.php

The results are as follows:

Test.php code is as follows:

<?php
class Test {
 public function hehe() {
 return &#39;hehe&#39;;
 }
 public function hehe2($params) {
 return json_encode($params);
 }
}

The directory structure is as follows:

Our customized protocol above can be modified at will, as long as the client and server can be unified and parsed.

The client passes the class, method and parameters to be called to the server by requesting the server, and the server returns the result by instantiating the calling method.

The above is the detailed content of Analyze how PHP can quickly create RPC services (code demonstration). For more information, please follow other related articles on the PHP Chinese website!

Statement:
This article is reproduced at:juejin.im. If there is any infringement, please contact admin@php.cn delete