共计 2845 个字符,预计需要花费 8 分钟才能阅读完成。
装置依赖
# composer.json
{
"require": {"php-amqplib/php-amqplib": ">=3.0"}
}
> composer.phar install
模式构造
RPC 工作流程:
- 当客户端启动时,会创立一个匿名独占回调队列。
-
对于 RPC 申请,客户端发送具备两个属性的音讯:
- reply_to,设置为回调队列;
- correlation_id,为每个申请设置惟一值。
- 申请被发送到 rpc_queue 队列。
- 服务器期待该队列上的申请。当申请呈现时,执行工作并将带有后果的音讯通过 reply_to 指定的队列发送回客户端。
- 客户端期待回调队列上的音讯。当呈现一条音讯时,查看 correlation_id 属性。如果它与申请中的值匹配,则将响应返回给应用程序。
客户端
<?php
require_once __DIR__ . '/vendor/autoload.php';
use PhpAmqpLib\Connection\AMQPStreamConnection;
use PhpAmqpLib\Message\AMQPMessage;
class FibonacciRpcClient
{
private $connection;
private $channel;
private $callback_queue;
private $response;
private $corr_id;
public function __construct()
{
// 创立连贯
$this->connection = new AMQPStreamConnection(
'localhost',
5672,
'guest',
'guest'
);
// 创立通道
$this->channel = $this->connection->channel();
// 创立队列,已存在的不会反复创立,第三个参数为敞开队列长久化
list($this->callback_queue, ,) = $this->channel->queue_declare(
"",
false,
false,
true,
false
);
// 第四个参数设为 false 敞开主动音讯确认,为 true 关上主动音讯确认即投递音讯后立即标记为删除
$this->channel->basic_consume(
$this->callback_queue,
'',
false,
true,
false,
false,
array(
$this,
'onResponse'
)
);
}
public function onResponse($rep)
{if ($rep->get('correlation_id') == $this->corr_id) {$this->response = $rep->body;}
}
public function call($n)
{
$this->response = null;
$this->corr_id = uniqid();
$msg = new AMQPMessage((string) $n,
array(
'correlation_id' => $this->corr_id,
'reply_to' => $this->callback_queue
)
);
// 通过默认的交换机发送音讯到队列 (音讯内容, 默认交换机, 路由键);
$this->channel->basic_publish($msg, '','rpc_queue');
while (!$this->response) {$this->channel->wait();
}
return intval($this->response);
}
}
$fibonacci_rpc = new FibonacciRpcClient();
// 发送 RPC 申请并阻塞直到收到响应
$response = $fibonacci_rpc->call(30);
echo '[.] Got', $response, "\n";
服务端
<?php
require_once __DIR__ . '/vendor/autoload.php';
use PhpAmqpLib\Connection\AMQPStreamConnection;
use PhpAmqpLib\Message\AMQPMessage;
// 创立连贯
$connection = new AMQPStreamConnection('localhost', 5672, 'guest', 'guest');
// 创立通道
$channel = $connection->channel();
// 创立队列,已存在的不会反复创立,第三个参数为敞开队列长久化
$channel->queue_declare('rpc_queue', false, false, false, false);
// 申明斐波那契生成函数
function fib($n)
{if ($n == 0) {return 0;}
if ($n == 1) {return 1;}
return fib($n-1) + fib($n-2);
}
echo "[x] Awaiting RPC requests\n";
// 定义回调函数
$callback = function ($req) {$n = intval($req->body);
echo '[.] fib(', $n, ")\n";
$msg = new AMQPMessage((string) fib($n),
array('correlation_id' => $req->get('correlation_id'))
);
$req->delivery_info['channel']->basic_publish(
$msg,
'',
$req->get('reply_to')
);
// 手动音讯确认
$req->ack();};
// 设置 prefetch_count = 1,开启偏心散发(默认为循环散发)// 在解决并确认上一条音讯之前,不要将新音讯发送给消费者,而发送给其余消费者
$channel->basic_qos(null, 1, null);
// 第四个参数设为 false 敞开主动音讯确认,为 true 关上主动音讯确认即投递音讯后立即标记为删除
$channel->basic_consume('rpc_queue', '', false, false, false, false, $callback);
while ($channel->is_open()) {$channel->wait();
}
$channel->close();
$connection->close();
运行
关上一个终端,运行服务端:
php rpc_server.php
# => [x] 期待 RPC 申请
关上另一个终端,运行服务端:
php rpc_server.php
# => [x] 期待 RPC 申请
关上另一个终端,运行客户端:
php rpc_client.php
# => [x] 申请 fib(30)
拓展性能
- 如果没有服务器在运行,客户端应该如何反馈?
- 客户端是否应该为 RPC 设置某种超时?
- 如果服务器呈现故障并引发异样,是否应该将其转发给客户端?
- 在解决之前避免有效的传入音讯(例如查看边界、类型)。
正文完