zoukankan      html  css  js  c++  java
  • 【原创】PHP访问MySQL查询超时处理

     

    【原创】PHP访问MySQL查询超时处理

    分类: Linux/FreeBSD/Server PHP/DHTML/Other2010-09-08 01:18 8382人阅读 评论(10) 收藏 举报

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

    手册里只有这么四个选项

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

    php-5.2.8/ext/mysqli/mysqli_api.c

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

    php-5.2.8/ext/mysqli/mysqli.c

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

    mysql-5.1.30/sql-common/client.c

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

    mysql-5.1.30/include/mysql.h

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

    mysql-5.1.30/sql-common/client.c

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

    mysql-5.1.30/sql/net_serv.cc

    现在基本得出了结论:

    按照上面查看代码来看,目前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";
    ?>

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

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

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

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

    MySQL相关的vio代码:

    poll 超时:

    setsockopt 超时:

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

  • 相关阅读:
    SQL SERVER 运维日记
    openstack
    Java GC 日志详解
    突破 BTrace 安全限制
    End-to-End Tracing of Ajax/Java Applications Using DTrace
    调试工具BTrace 的使用--例子
    btrace-dtrace-for-java-ish
    DTrace Probes in HotSpot VM
    DTrace memory leak 内存泄露
    Java-JVM-GC
  • 原文地址:https://www.cnblogs.com/persist/p/3515493.html
Copyright © 2011-2022 走看看