PHP使用Redis长连接的方法详解
本篇章节讲解PHP使用Redis长连接的方法。分享给大家供大家参考,具体如下: php-redis在github上的项目地址:https://github.com/phpredis/phpredis pconnect函数声明 其中time_out表示客户端闲置多少秒后,就断开连接。函数连接成功返回true,失败返回false: 下面的例子详细介绍了pconnect连接的重用情况。 pconnect('127.0.0.1',6379);
$redis->pconnect('127.0.0.1'); // 默认端口6379,跟上面的例子使用相同的连接。
$redis->pconnect('127.0.0.1',6379,2.5); // 设置了2.5秒的过期时间。将是不同于上面的新连接
$redis->pconnect('127.0.0.1',2.5,'x'); //设置了持久连接的id,将是不同于上面的新连接
$redis->pconnect('/tmp/redis.sock'); // unix domain socket - would be another connection than the four before.
pconnect使用介绍 对pconnect方法简单描述使用该方法创建连接,连接不会在调用close方法之后关闭,只有在进程结束之后该连接才会被关闭。
长连接只会在PHP-FPM进程结束之后结束,连接的生命周期就是PHP-FPM进程的生命周期。 相比较短连接而言,在每一个PHP-FPM调用过程中都会产生一个redis的连接,在服务器上的表性形式就是过多的time_out连接状态。 而长连接相反,PHP-FPM调用的所有CGI都只会共用一个长连接,所以也就是只会产生固定数量的time_out。 关闭长连接可以调用close和unset方法,但两则差异很大: -
-
等效的问题是,在单例模式中,判断当前实例是否有效。 习惯上调用echo,判断是否正常返回字符串本身,或者调用ping,查看返回值是否为 +PONG。 但是需要特别小心的是,在redis断开连接之后,调用echo以及ping(返回'+POMG')时,均会抛出异常。所以要通过异常捕获机制来处理。 代码分析pconnect连接重用的问题 说明:a实例和b实例共用了一条连接,b实例将a实例的连接修改了: 所以下面的例子导致最终$a实例得到的值变成了2,需要特别注意。 setex(id,3);
echo $a -> get(id);
//之后执行下面的连接
$b = pconnect(host,time_out);
select(2);
$b->set(id,2)
echo $a->get(id); //这个id操作的db变成了2,不再是之前的3了。因为这两个连接共用了一个连接通道。
情况二:单例模式。将上述的代码修改,a和b都通过getInstance来生成。生成的前提是判断当前实例是否存在。单例模式的混淆点在于: $a生成了一个实例,这时候生成$b,$b使用了$a的实例,然后修改了$a的连接,之后调用$a肯定是调用的$b修改之后的实例。跟情况二一致。 单例模式的代码如下: select($db);
return self::$_instance;
}
两种情况都说明了连接重用的问题。如何修复这个bug?两点: 1.为每一个db生成一个单例。 2.避免连接重用问题。 所以代码可以做调整为返回一个单例数组: Ping() == 'Pong') {
return self::$_instance[$db];
}
} catch (Exception $e) {
}
self::$_instance[$db] = new Redis();
self::_connect($db);
return self::$_instance[$db];
}
需要注意的地方 避免在Task类成员变量中使用redis对象。 在redis的单例模式中,声明了time_out的过期时间。如果redis处理的场合是一个任务,而任务调用redis间隔时间又比较长。当间隔大于time_out时候,redis就会断开连接,这时候所有对redis的操作都会失效。解决的办法就是避免这种调用方式,通过在调用的地方动态声明redis类来执行。这种问题对于长连接和短链接是没有区分,属于调用的方式错误。 更多关于PHP相关内容感兴趣的读者可查看本站专题:《》、《》、《》、《》、《》、《》及《》 希望本文所述对大家PHP程序设计有所帮助。 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |