源码网商城,靠谱的源码在线交易网站 我的订单 购物车 帮助

源码网商城

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

  • 时间:2021-06-21 21:24 编辑: 来源: 阅读:
  • 扫一扫,手机访问
摘要:PHP访问MySQL查询超时处理的方法
目前两个客户端扩展库连接超时可以设置选项来操作,比如mysqli:
[u]复制代码[/u] 代码如下:
<?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手册没有发现这个操作选项或者函数。   手册里只有这么四个选项 [img]http://files.jb51.net/upload/201105/20110502131947724.jpg[/img] 跟踪 mysqli 的扩展源代码发现它底层调用的是 libmysqlclient 的 mysql_options: php-5.2.8/ext/mysqli/mysqli_api.c [img]http://files.jb51.net/upload/201105/20110502131947144.jpg[/img] 并且在mysqli的PHP扩展中就只导出了几个变量: php-5.2.8/ext/mysqli/mysqli.c [img]http://files.jb51.net/upload/201105/20110502131947336.jpg[/img] 大概看了一下 libmysqlclient 的代码,发现其实它自带是有读写超时设置的: mysql-5.1.30/sql-common/client.c [img]http://files.jb51.net/upload/201105/20110502131947467.jpg[/img] 因为它自己定义了很多操作选项,只是php扩展里没有: mysql-5.1.30/include/mysql.h [img]http://files.jb51.net/upload/201105/20110502131947976.jpg[/img] 看看mysql中的读写超时是如何实现的: mysql-5.1.30/sql-common/client.c [img]http://files.jb51.net/upload/201105/20110502131948608.jpg[/img] [img]http://files.jb51.net/upload/201105/20110502131948736.jpg[/img] 读写超时真正操作的地方,超时处理这里重试了两次,还是写死了: mysql-5.1.30/sql/net_serv.cc [img]http://files.jb51.net/upload/201105/20110502131948859.jpg[/img] [img]http://files.jb51.net/upload/201105/20110502131948280.jpg[/img] 现在基本得出了结论:

按照上面查看代码来看,目前PHP针对MySQL查询超时以下限制: 1. 超时设置单位为秒,最少配置1秒 2. 但mysql底层的read会重试两次,所以实际会是 3 秒   重试两次 + 自身一次 = 3倍超时时间。   那么就是说最少超时时间是3秒,不会低于这个值,对于大部分应用来说可以接受,但是对于小部分应用需要优化。 现在我们来看看如果我们自己要设置超时,我们自己压入 MYSQL_OPT_READ_TIMEOUT 也是可以达到读写超时效果的,写一段代码来测试一下:

[u]复制代码[/u] 代码如下:
<?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"; ?>
查看上面代码的执行结果,验证了上面的观点,第一个查询成功了,第二个查询连接被断开了: [img]http://files.jb51.net/upload/201105/20110502131948838.jpg[/img] 如果需要修改这个秒级别的超时,比如改成毫秒级别的超时,只能两个地方修改:   1.  修改客户端,比如 mysqli query 代码,加入定时器,超时则返回 2.  修改 Mysql 中的vio代码,因为mysql的网络处理底层都是经过vio的操作 MySQL相关的vio代码: poll 超时: [img]http://files.jb51.net/upload/201105/20110502131948499.jpg[/img] setsockopt 超时: [img]http://files.jb51.net/upload/201105/20110502131948206.jpg[/img] 基本上到这里就基本能够解决PHP在针对MySQL读写查询操作超时的处理了,希望对你有帮助。 heiyeluren的blog

  • 全部评论(0)
联系客服
客服电话:
400-000-3129
微信版

扫一扫进微信版
返回顶部