PHP访问MySQL查询超时处理的方法

时间:2023-11-23 03:05:48 

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


<?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手册没有发现这个操作选项或者函数。

 

手册里只有这么四个选项

PHP访问MySQL查询超时处理的方法

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

php-5.2.8/ext/mysqli/mysqli_api.c

PHP访问MySQL查询超时处理的方法

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

php-5.2.8/ext/mysqli/mysqli.c

PHP访问MySQL查询超时处理的方法

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

mysql-5.1.30/sql-common/client.c

PHP访问MySQL查询超时处理的方法

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

mysql-5.1.30/include/mysql.h

PHP访问MySQL查询超时处理的方法

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

mysql-5.1.30/sql-common/client.c

PHP访问MySQL查询超时处理的方法

PHP访问MySQL查询超时处理的方法

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


mysql-5.1.30/sql/net_serv.cc

PHP访问MySQL查询超时处理的方法




PHP访问MySQL查询超时处理的方法




现在基本得出了结论:


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

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

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

 

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

 

那么就是说最少超时时间是3秒,不会低于这个值,对于大部分应用来说可以接受,但是对于小部分应用需要优化。
现在我们来看看如果我们自己要设置超时,我们自己压入 MYSQL_OPT_READ_TIMEOUT 也是可以达到读写超时效果的,写一段代码来测试一下:


<?php
//自己定义读写超时常量
if (!defined('MYSQL_OPT_READ_TIMEOUT')) {
define('MYSQL_OPT_READ_TIMEOUT', 11);
}
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";
?>


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


PHP访问MySQL查询超时处理的方法


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

 

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

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



MySQL相关的vio代码:


poll 超时:

PHP访问MySQL查询超时处理的方法


setsockopt 超时:


PHP访问MySQL查询超时处理的方法



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

标签:查询超时
0
投稿

猜你喜欢

  • YOLOv5车牌识别实战教程(三)模型训练与评估

    2021-01-07 19:15:43
  • python 制作本地应用搜索工具

    2023-03-25 02:16:50
  • 详解如何模拟实现node中的Events模块(通俗易懂版)

    2024-05-05 09:20:55
  • python 通过dict(zip)和{}的方式构造字典的方法

    2023-10-03 00:05:12
  • python脚本框架webpy模板控制结构

    2023-07-08 23:45:34
  • 非常详细的SQL--JOIN之完全用法

    2024-01-17 11:59:19
  • TensorFlow设置日志级别的几种方式小结

    2023-07-30 07:57:42
  • Python selenium爬取微博数据代码实例

    2023-07-01 02:46:49
  • 页面嵌入Windows Media Player播放器代码需要注意的

    2023-07-02 17:04:48
  • mysql实现事务的提交和回滚实例

    2024-01-23 04:13:12
  • python中return不返回值的问题解析

    2021-04-26 20:36:07
  • python3+PyQt5实现支持多线程的页面索引器应用程序

    2022-02-17 02:02:11
  • 彻底搞懂Python字符编码

    2023-10-14 01:05:03
  • python结合shell查询google关键词排名的实现代码

    2023-10-11 21:28:47
  • 关于MySQL中explain工具的使用

    2024-01-18 01:51:15
  • Python爬虫练习汇总

    2023-04-27 03:16:37
  • mysql 数据库备份的多种实现方式总结

    2024-01-18 23:03:25
  • 编写Python脚本把sqlAlchemy对象转换成dict的教程

    2023-02-22 07:26:47
  • golang之数组切片的具体用法

    2024-04-29 13:06:43
  • SpringBoot配置数据库密码加密的实现

    2024-01-26 07:33:01
  • asp之家 网络编程 m.aspxhome.com