溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

PHP實現(xiàn)遠(yuǎn)程過程調(diào)用RPC

發(fā)布時間:2020-06-21 02:14:48 來源:網(wǎng)絡(luò) 閱讀:1893 作者:hgditren 欄目:web開發(fā)
一、初識RPC

RPC(Remote Procedure Call)—遠(yuǎn)程過程調(diào)用,它是一種通過網(wǎng)絡(luò)從遠(yuǎn)程計算機程序上請求服務(wù),而不需要了解底層網(wǎng)絡(luò)技術(shù)的協(xié)議。

二、工作原理

運行時,一次客戶機對服務(wù)器的RPC調(diào)用,其內(nèi)部操作大致有如下十步:

  • 1.調(diào)用客戶端句柄;執(zhí)行傳送參數(shù)
  • 2.調(diào)用本地系統(tǒng)內(nèi)核發(fā)送網(wǎng)絡(luò)消息
  • 3.消息傳送到遠(yuǎn)程主機
  • 4.服務(wù)器句柄得到消息并取得參數(shù)
  • 5.執(zhí)行遠(yuǎn)程過程
  • 6.執(zhí)行的過程將結(jié)果返回服務(wù)器句柄
  • 7.服務(wù)器句柄返回結(jié)果,調(diào)用遠(yuǎn)程系統(tǒng)內(nèi)核
  • 8.消息傳回本地主機
  • 9.客戶句柄由內(nèi)核接收消息
  • 10.客戶接收句柄返回的數(shù)據(jù)
    (以上信息來自百度百科)
三、在PHP中的實現(xiàn)
  • 文件結(jié)構(gòu)
zrj@zrj:~/www/zhangrenjie_test/test/rpc$ tree
├── Client.php
└── Provider
    ├── index.php
    └── Services
        └── UserService.php
  • 客戶端Client.php
class Client
{
    private $serviceUrl;
    private $serviceName;
    private $rpcConfig = [
        'UserService' => 'http://127.0.0.1:8081',
    ];

    public function __construct($serviceName)
    {
        if (array_key_exists($serviceName, $this->rpcConfig)) {
            $this->serviceUrl = $this->rpcConfig[$serviceName];
            $this->serviceName = $serviceName;
        }
    }

    // __call() is triggered when invoking inaccessible methods in an object context.
    //調(diào)用不可訪問的方法
    public function __call($actionName, $arguments)
    {
        $content = json_encode($arguments);
        $options['http'] = [
            'timeout' => 5,
            'method' => 'POST',
            'header' => 'Content-type:applicaion/x-www-form-urlencoode',
            'content' => $content,
        ];
        //創(chuàng)建資源流上下文
        $context = stream_context_create($options);

        $get = [
            'service_name' => $this->serviceName,
            'action_name' => $actionName
        ];

        $serviceUrl = $this->serviceUrl . '?' . http_build_query($get);
        $result = file_get_contents($serviceUrl, false, $context);
        return json_decode($result, true);
    }
}

$userService = new Client('UserService');
$result=$userService->getUserInfo(random_int(1,10));
var_dump($result);

步驟:

  • 1.注冊服務(wù)$rpcConfig。
  • 2.實例化客戶端時,指定服務(wù)名稱。
  • 3.訪問對象不可訪問的方法getUserInfo,自動調(diào)用__call魔術(shù)方法。
  • 4.在__call魔術(shù)方法中想指定的遠(yuǎn)程服務(wù)發(fā)送請求(http://127.0.0.1:8081)
  • 服務(wù)提供者(Service Provider)的實現(xiàn)

Provider目錄為演示的服務(wù)提供者的總目錄,index.php為管理調(diào)度服務(wù)的入口。
Provider\Services目錄為所有服務(wù)類目錄。

服務(wù)調(diào)度入口:index.php

namespace rpc\Provider;

require_once './Services/UserService.php';

use rpc\Provider\Services\{
    UserService
};

$serviceName = trim($_GET['service_name']);
$serviceAction = trim($_GET['action_name']);
$argv = file_get_contents("php://input");
if (empty($serviceName) || empty($serviceAction)) die('paramas is missing');
if (!empty($argv)) {
    $argv = json_decode($argv, true);
}

$result = call_user_func_array([$serviceName, $serviceAction], $argv);
echo  json_encode($result);

User服務(wù):UserService.php


namespace rpc\Provider\Services;

class UserService
{
    public static function getUserInfo(int $uid): array
    {
        return [
            'id' => $uid,
            'user_name' => 'jack_' . $uid,
        ];
    }
}

開啟服務(wù):對外提供服務(wù)遠(yuǎn)程調(diào)用地址(http://127.0.0.1:8081)。

#這里我們利用PHP自帶的cli模式開啟服務(wù)
php -S 127.0.0.1:8080 -t /www/zhangrenjie_test/test/rpc/Provider

至此,大功告成。

了解php開啟服務(wù),請參照沒有第三方web服務(wù),怎么運行php?

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI