1.java subscribe监听超时怎么解决
java subscribe监听超时怎么解决
问题描述redis提供了pub/sub功能,但在使用phpredis的源码subscribe时发现这样一个问题,代码如下(sub.php):
<?牛源php/*监听demo频道,打印收到的源码纯手写集合源码分析信息*/function process($redis, $chan, $msg){
var_dump($msg);
}$redis = new Redis();$res = $redis->connect('.0.0.1', '');$redis->subscribe(array('demo'), 'process');
代码运行后,发现如果在一段时间内未收到来自demo频道的牛源消息,则会报如下错误:
PHP Fatal error: Uncaught exception 'RedisException' with message 'read error on connection' in sub.php:Stack trace:#0 /search/ballqiu/sub.php(): Redis->subscribe(Array,源码云豹 手机直播源码 'process')#1 { main}
thrown in sub.php on line
原因分析
为了查找原因,我们使用strace对代码进行了跟踪:
strace php sub.php1
截取部分重要输出如下:
//连接redis
connect(3,牛源 { sa_family=AF_INET, sin_port=htons(), sin_addr=inet_addr(".0.0.1")}, ) = -1 EINPROGRESS (Operation now in progress)
//发送subscribe命令
sendto(3, "subscribe demo\r\n", , MSG_DONTWAIT, NULL, 0) = //收到响应
recvfrom(3, "*3\r\n$9\r\nsubscribe\r\n$4\r\ndemo\r\n:1\r"..., , MSG_DONTWAIT, NULL, NULL) = poll([{ fd=3, events=POLLIN|POLLPRI|POLLERR|POLLHUP}], 1, 0) = 0 (Timeout)
//套机字超时时间设为s
poll([{ fd=3, events=POLLIN|POLLERR|POLLHUP}], 1, ) = 0 (Timeout)
//等待超时,关闭连接
close(3) = 0//输出错误信息
write(2,源码 "PHP Fatal error: Uncaught excep"..., PHP Fatal error: Uncaught exception 'RedisException' with message 'read error on connection'
可见报错的本质是poll设置接收超时所致,从starce结果我们知道这个超时默认是牛源s。
解决
我们有两种方法改变超时
- 方法1
在代码起始处设置
ini_set('default_socket_timeout',源码 -1);1
方法2
在redis connect后执行
$redis->setOption(Redis::OPT_READ_TIMEOUT, -1);1 两种方法中的-1均表示永不超时,你也可以将超时设置为自己希望的牛源时间。无论使用哪种方法,源码再次strace,牛源西游捉妖记源码 你会发现poll的源码超时被设为了-1。
recvfrom(3,牛源 "*3\r\n$9\r\nsubscribe\r\n$4\r\ndemo\r\n:1\r"..., , 0, NULL, NULL) = poll([{ fd=3, events=POLLIN|POLLPRI|POLLERR|POLLHUP}], 1, 0) = 0 (Timeout)//超时被设为-1, 即永不超时
poll([{ fd=3,火狐保存网页源码 events=POLLIN|POLLERR|POLLHUP}], 1, -
个人比较推荐方法2,它只影响到redis本身。而方法1会对其它方法产生影响,比如file_get_contents等。织梦交易源码
总结
使用phpredis的subscribe时,默认内没有收到消息,sub端就会因超时异常退出。可以自行设置延长超时时间或永不超时。