服务器之家:专注于服务器技术及软件下载分享
分类导航

Mysql|Sql Server|Oracle|Redis|

服务器之家 - 数据库 - Mysql - PHP访问MySQL查询超时处理的方法

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

2019-11-18 15:11heiyeluren的blog Mysql

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

目前两个客户端扩展库连接超时可以设置选项来操作,比如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

延伸 · 阅读

精彩推荐
  • MysqlMySQL数据库十大优化技巧

    MySQL数据库十大优化技巧

    WEB开发者不光要解决程序的效率问题,对数据库的快速访问和相应也是一个大问题。希望本文能对大家掌握MySQL优化技巧有所帮助。 ...

    mysql教程网4032019-11-15
  • MysqlMysql如何避免全表扫描的方法

    Mysql如何避免全表扫描的方法

    如果MySQL需要做一次全表扫描来处理查询时,在 EXPLAIN 的结果中 type 字段的值是 ALL。 ...

    mysql教程网2702019-10-24
  • Mysqlmysql 查看版本的方法图文演示

    mysql 查看版本的方法图文演示

    今天打算升级下mysql数据库,可不知道现在的版本是多少,从网上找了一些资料,发现还是这些好用。 ...

    mysql教程网1742019-11-07
  • MysqlMySQL InnoDB和MyISAM数据引擎的差别分析

    MySQL InnoDB和MyISAM数据引擎的差别分析

    InnoDB和MyISAM是在使用MySQL最常用的两个表类型,各有优缺点,视具体应用而定。基本的差别为:MyISAM类型不支持事务处理等高级处理,而InnoDB类型支持 ...

    mysql教程网1212019-11-18
  • Mysql一键安装mysql5.7及密码策略修改方法

    一键安装mysql5.7及密码策略修改方法

    这篇文章主要介绍了一键安装mysql5.7及密码策略修改方法,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下...

    旅行者-Travel3082019-06-20
  • Mysql分析Mysql表读写、索引等操作的sql语句效率优化问题

    分析Mysql表读写、索引等操作的sql语句效率优化问题

    今天小编就为大家分享一篇关于分析Mysql表读写、索引等操作的sql语句效率优化问题,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需...

    执笔记忆的空白3382019-06-13
  • MysqlMySQL8新特性:持久化全局变量的修改方法

    MySQL8新特性:持久化全局变量的修改方法

    这篇文章主要给大家介绍了关于MySQL 8新特性:持久化全局变量的修改的相关内容,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参...

    iVictor4592019-07-02
  • Mysql使用mysql中遇到的几个问题

    使用mysql中遇到的几个问题

    首先mysql不是可视化的,可以通过命令行进行操作,包括创建数据库、表、添加数据等等。那岂不是很不方便了吗? ...

    mysql教程网4232019-10-30