在 OpenMix 全家桶 中有一个 Mix Redis Subscribe 的项目,这是一个不依赖 phpredis 扩展,直接解析 Redis 协议专用于订阅处理的一个库,任何 Swoole 框架均可使用,可普遍使用于 WebSocket 开发中,在 MixPHP 骨架中也默认包含了这个库。php
MixPHP V2.1 完成开发后,我试图开发一个基于订阅机制的 WebScoket 服务,该服务须要可动态切换订阅频道,但 phpredis 的订阅方法没法实现如下功能:git
$redis = new \Redis(); $res = $redis->pconnect('127.0.0.1', 6379, 0); $redis->subscribe(['test'], function ($instance, $channelName, $message) { echo $channelName, "==>", $message, PHP_EOL; });
以上代码中,当执行到 subscribe 会阻塞执行,只有在有消息过来时才会执行到匿名函数中,并不会在订阅成功的当时执行该闭包,可是 redis-cli 执行订阅时,redis-server 是有回复订阅成功消息的,所以是 phpredis 的设计问题。github
因为 subscribe 阻塞了执行,代码只能在有消息触发回调时才能在回调中执行,所以动态增长频道也是没法操做的。golang
由于上面那种阻塞回调的设计,若是须要取消一个频道,只能在有消息过来时方可操做,可是实际需求一般是须要在任意时刻均可取消频道。redis
phpredis 当试图在匿名函数之外的其余协程中 close 链接会抛出异常 PHP Fatal error: Uncaught RedisException: read error on connection
,这让关闭一个订阅中的 redis 链接都没法优雅的实现。安全
当我得知 redis 协议是简单的文本协议时,我决定抛弃 phpredis 本身造一个好用的订阅库,新轮子具备如下优势:闭包
$sub = new \Mix\Redis\Subscribe\Subscriber([ // 链接失败将抛出异常 'host' => '192.168.198.1', 'port' => 6379, 'timeout' => 5, 'password' => '', ]); $sub->subscribe('foo', 'bar'); // 订阅失败将抛出异常 $chan = $sub->channel(); while (true) { $data = $chan->pop(); if (empty($data)) { // 手动close与redis异常断开都会致使返回false if (!$sub->closed) { // redis异常断开处理 var_dump('Redis connection is disconnected abnormally'); } break; } var_dump($data); }
接收到订阅消息:框架
object(Mix\Redis\Subscribe\Message)#8 (2) { ["channel"]=> string(2) "foo" ["payload"]=> string(4) "test" }