PHP- 深入PHP、Redis连接
pconnect, phpredis中用于client连接server的api。
The connection will not be closed on close or end of request until the php process ends.
这是api说明中的一句原文
那么问题来了:
1. php process ends是指一次php执行完结,还是fpm的终结?如果是后者,那意味着即使一次php执行完毕,redis连接也不会被释放,下一次执行时redis连接会被重用。
2. The connection will not be closed on close是 说如果使用了pconnect, 即使在代码中显示的调用close(), 也不会关闭连接?
带着这两个问题,我们做下实验,深入看一下pconnect究竟做了些什么。
准备工作
环境:
nginx + php-fpm
php5.3
我们将fpm配置为
- pm.max_children =
- pm.start_servers =
- pm.max_spare_servers =
这样,我们的页面请求会由一个确定的fpm进程执行,方便strace跟踪。
对应页面请求的php代码:
- $ip = "10.136.30.144";
- $port = 7777;
- $redis = new Redis();
- $redis->pconnect($ip, $port, 1);
- $key = "test";
- $value = "this is test";
- $redis->set($key, $value);
- $d = $redis->get($key);
- var_dump($d);
代码的功能很简单,连接redis,先设置一个值,再取出。
测试问题1
思路:
使用strace观察fpm的系统调用,如果连接的生命周期是一次php执行,那么每次页面调用,都会有connect系统调用,用以连接redis;如果连接的生命周期是fpm的终结,那么只有第一次页面调用会有connect系统调用 ,之后由于连接被重用,无需connect,直接发命令请求即可。
启动一个新的fpm(进程号28082)。
执行
- strace -p -s -o redis_1
记录一次页面请求的系统调用。如下图所示:
可以看到进程先建立了socket连接(文件描述符为9)。然后给reids发一系列命令,最终取到“this is test”的结果串。且没有关闭连接相关的redis命令或系统调用。
页面请求结束后,我们执行
- lsof -n -p
可以看到,fpm进程仍然保有一个到10.136.30.144的reids连接,其文件描述符为9(这与strace的结果一致)。
执行
- strace -p -s -o redis_2
记录 第二次页面请求的系统调用,得到下面结果。
与第一次请求的区别是:省去了建立连接的过程,直接发送reids命令,得到结果!
再使用lsof -n -p 28082查看fpm打开的文件描述符,结果与上文件相同。
说明,连接的确是被重用的,没有新建。
执行第6次页面请求(因为我们在准备工作中的配置,此时fpm已经是一个新的进程了),用lsof查看进程打开的文件描述符。 我们发现,虽然仍然有描述符为9的reids连接,但两个tcp连接的临时端口不同了,也就是连接改变了!
至此,我们得出问题1的结论:
当使用pconnect时,连接会被重用,连接的生命周期是fpm进程的生命周期,而非一次php的执行。。
测试问题2
为了对比,我们先看一下,使用connect连接redis,并调用redis->close()的系统调用。(将上述代码中的pconnect改为connect, 同时在最后加入redis->close())
我们看到,除了建立连接外,在程序结尾,还向reids发送了quit命令,并关闭了连接的文件描述符。
接下来,我们看在使用pconnect后,redis->close()有何表现
代码调整为:
- $ip = "10.136.30.144";
- $port = ;
- $redis = new Redis();
- $redis->pconnect($ip, $port, );
- $key = "test";
- $value = "this is test";
- $redis->set($key, $value);
- $d = $redis->get($key);
- var_dump($d);
- $redis->close();
- try {
- $redis->get($key);
- } catch (Exception $e) {
- echo $e->getMessage();
- }
我们直接看第2次执行页面请求的系统调用
并没有建立连接,同样是直接发送命令得到结果。说明连接被重用。同时,没有向reids server发送quit命令,也无关闭连接的系统调用。
但注意,页面请求的返回结果:
至此,我们得出问题2的结论:
如果代码中使用pconnect, close的作用仅是使当前php不能再进行redis请求,但无法真正关闭redis长连接,连接在后续请求中仍然会被重用,直至fpm进程生命周期结束。
结论
1. 当使用pconnect时,连接会被重用,连接的生命周期是fpm进程的生命周期,而非一次php的执行。
2.如果代码中使用pconnect, close的作用仅是使当前php不能再进行redis请求,但无法真正关闭redis长连接,连接在后续请求中仍然会被重用,直至fpm进程生命周期结束。
转自:http://blog.csdn.net/qmhball/article/details/46988111
PHP- 深入PHP、Redis连接的更多相关文章
- Redis 连接问题
.NET 中使用 StackExchange.Redis 我为什么想写这个,总感觉很多介绍相应技术的博客,只是把内容从官网搬到自己的博客中,没有任何的实践,这样会给想学的人,没有任何好处,也可能我是自 ...
- Redis 连接池的问题
目录 Redis 连接池的问题 1 1. 前言 1 2.解决方法 1 前言 问题描述:Redis跑了一段时间之后,出现了以下异常. Redis Timeout ex ...
- Redis 连接
Redis 连接命令主要是用于连接 redis 服务. 实例 以下实例演示了客户端如何通过密码验证连接到 redis 服务,并检测服务是否在运行: redis 127.0.0.1:6379> ...
- Python连接Redis连接配置
1. 测试连接: Python 2.7.8 (default, Oct 20 2014, 15:05:19) [GCC 4.9.1] on linux2 Type "help", ...
- 2016022613 - redis连接命令集合
redis连接命令 1.ping 用途:检查服务器是否正在运行 返回数据pong,表示服务器在运行. 2.quit 用途:关掉当前服务器连接 3.auth password 用途:服务器验证密码 没有 ...
- 红眼技术博客 » redis连接池红眼技术博客 » redis连接池
红眼技术博客 » redis连接池 redis连接池
- redis连接池操作
/** * @类描述 redis 工具 * @功能名 POJO * @author zxf * @date 2014年11月25日 */public final class RedisUtil { p ...
- java操作redis redis连接池
redis作为缓存型数据库,越来越受到大家的欢迎,这里简单介绍一下java如何操作redis. 1.java连接redis java通过需要jedis的jar包获取Jedis连接. jedis-2.8 ...
- 三:Redis连接池、JedisPool详解、Redisi分布式
单机模式: package com.ljq.utils; import redis.clients.jedis.Jedis; import redis.clients.jedis.JedisPool; ...
随机推荐
- (转)MyEclipse设置注释格式
原文:http://xinghaifeng2006.iteye.com/blog/1243565 MyEclipse设置注释格式(转载) 博客分类: Java基础知识 Windo ...
- META标签的NAME变量
META标签的NAME变量语法格式是: <META NAME=xxx CONTENT=xxxxxxxxxxxxxxxxxx> 其中xxx主要有下面几种参数: 1. Keywords(关键字 ...
- C++面向对象编程解决三阶矩阵相加减
/*此处用面向对象编程*/ #include<iostream> #include<string.h> using namespace std; class Matrices ...
- 删除目录下所有gif的图片
find -name "*.gif" -exec rm -fv {} \;
- POJ 2436 二进制枚举+位运算
题意:给出n头牛的得病的种类情况,一共有m种病,要求找出最多有K种病的牛的数目: 思路:二进制枚举(得病处为1,否则为0,比如得了2 1两种病,代号就是011(十进制就是3)),首先枚举出1的个数等于 ...
- CMAKE语法
[语法] 注释 # : #我是注释 命令语法 COMMAND: COMMAND(参数1 参数2 ...) 字符串列 A;B;C //分号分割或空格分隔的值 变量 ...
- Linux systemd 打开调试终端、添加开机自运行程序
/************************************************************************* * Linux systemd 打开调试终端.添加 ...
- php计算一天的时间
代码如下: $time=time(); //一天的开始 $start=strtotime(date('Y-m-d',$time).'00:00:01'); echo date('Y-m-d H:i:s ...
- 《JavaScript Ninja》之挥舞函数
挥舞函数 匿名函数为什么如此重要 通常使用匿名函数的情况是,创建一个供以后使用的函数.例如,将匿名函数保存在一个变量里,将其作为一个对象的方法,或者是将匿名函数作为一个回调.-->在这些情况下, ...
- 解决chrome同步问题
原方法为 修改文件:“C:\WINDOWS\system32\drivers\etc\hosts”,但是经常失败,需要重新修改,很是麻烦. 更好的方法是使用 “谷歌访问助手(chrome版)” 具体参 ...