php mysql 查询超时_PHP访问MySQL查询超时处理

PHP连接MySQL主要是使用Mysql提供的 libmysqlclient 的客户端库,同时也延伸出来 mysql 和  mysqli 两套PHP的扩展,相对来说 mysqli 比 mysql 更好,更稳定。

目前两个客户端扩展库连接超时可以设置选项来操作,比如mysqli:

8f900a89c6347c561fdf2122f13be562.png

<?php //创建对象$mysqli=mysqli_init();//设置超时选项$mysqli->options(MYSQLI_OPT_CONNECT_TIMEOUT,5);//连接$mysqli->real_connect('localhost','my_user','my_password','world');//如果超时或者其他连接失败打印错误信息if(mysqli_connect_errno()) {printf("Connect failed: %s\n",mysqli_connect_error());exit();

}//成功输出连接信息printf("Connection: %s\n.",$mysqli->host_info);$mysqli->close();?>

这个是连接超时,但是有些时候我们需要查询读写超时,比如说我们一个数据库压力很大,或者连接很多,那么数据库查询就很缓慢,但是我希望某些不重要的数据,比如说文章点击数这种如果查询超时了就不显示,至少能够保证主体页面正确显示,但是查遍PHP手册没有发现这个操作选项或者函数。

手册里只有这么四个选项

6246594d4d3e2d35eb60ec640daf7311.png

跟踪 mysqli 的扩展源代码发现它底层调用的是 libmysqlclient 的 mysql_options:

php-5.2.8/ext/mysqli/mysqli_api.c

f6880635c3a178995f6490948a64607a.png

并且在mysqli的PHP扩展中就只导出了几个变量:

php-5.2.8/ext/mysqli/mysqli.c

e602bcbecc4be157cc37ca4030911e90.png

大概看了一下 libmysqlclient 的代码,发现其实它自带是有读写超时设置的:

mysql-5.1.30/sql-common/client.c

5443da9ebe9b9f68de929003609f1e2c.png

因为它自己定义了很多操作选项,只是php扩展里没有:

mysql-5.1.30/include/mysql.h

5720b2e002588c09800342a2cc9dcfe2.png

看看mysql中的读写超时是如何实现的:

mysql-5.1.30/sql-common/client.c

a95a61a48cfc8321cd11a6f428101527.png

985aa1529ef46b2b11c5d85e567c59d5.png

读写超时真正操作的地方,超时处理这里重试了两次,还是写死了:

mysql-5.1.30/sql/net_serv.cc

2bc4233169e116e27665f100e67c4905.png

6ef2303a5cc9910f8cc5bfea2413b844.png

现在基本得出了结论:

按照上面查看代码来看,目前PHP针对MySQL查询超时以下限制:

1. 超时设置单位为秒,最少配置1秒

2. 但mysql底层的read会重试两次,所以实际会是 3 秒

重试两次 + 自身一次 = 3倍超时时间。

那么就是说最少超时时间是3秒,不会低于这个值,对于大部分应用来说可以接受,但是对于小部分应用需要优化。

现在我们来看看如果我们自己要设置超时,我们自己压入 MYSQL_OPT_READ_TIMEOUT 也是可以达到读写超时效果的,写一段代码来测试一下:

8f900a89c6347c561fdf2122f13be562.png

}if(!defined('MYSQL_OPT_WRITE_TIMEOUT')) {define('MYSQL_OPT_WRITE_TIMEOUT',12);

}//设置超时$mysqli=mysqli_init();$mysqli->options(MYSQL_OPT_READ_TIMEOUT,3);$mysqli->options(MYSQL_OPT_WRITE_TIMEOUT,1);//连接数据库$mysqli->real_connect("localhost","root","root","test");if(mysqli_connect_errno()) {printf("Connect failed: %s\n",mysqli_connect_error());exit();

}//执行查询 sleep 1秒不超时printf("Host information: %s\n",$mysqli->host_info);if(!($res=$mysqli->query('select sleep(1)'))) {echo"query1 error:".$mysqli->error."\n";

}else{echo"Query1: query success\n";

}//执行查询 sleep 9秒会超时if(!($res=$mysqli->query('select sleep(9)'))) {echo"query2 error:".$mysqli->error."\n";

}else{echo"Query2: query success\n";

}$mysqli->close();echo"close mysql connection\n";?>

查看上面代码的执行结果,验证了上面的观点,第一个查询成功了,第二个查询连接被断开了:

09923af4decf031e2f33bdf7d32f33b8.png

如果需要修改这个秒级别的超时,比如改成毫秒级别的超时,只能两个地方修改:

1.  修改客户端,比如 mysqli 的 query 代码,加入定时器,超时则返回

2.  修改 Mysql 中的vio代码,因为mysql的网络处理底层都是经过vio的操作

MySQL相关的vio代码:

poll 超时:

811f9cb72c84b8773e14b4f1d0e64891.png

setsockopt 超时:

7ddb87b1e648c451df6a203c9d46d9de.png

基本上到这里就基本能够解决PHP在针对MySQL读写查询操作超时的处理了,希望对你有帮助。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值