swoole结合php的pdo mysql模式出现MySQL server has gone away

参考:   https://my.oschina.net/fage1151/blog/995775                常驻内存的PHP程序mysql报错 mysql has gone away

              https://www.cnblogs.com/xieyong-p/p/9808246.html     Yii2 框架解决 MySQL server has gone away问题

 

原因

    这要从mysql的机制说起,mysql本身是一个多线程的程序,每个连接过来,会开一个线程去处理相关的query, 所以mysql为了避免占着毛坑不拉屎,会定期回收长时间没有任何query的连接(时间周期受wait_timeout配置影响),所以在swoole中,由于是一个长驻内存的服务,我们建立了一个mysql的连接,不主动关闭 或者是用pconnect的方式,那么这个mysql连接会一直保存着,然后长时间没有和数据库有交互,就主动被mysql server关闭了,之后继续用这个连接,就报mysql server gone away了。

 

解决方案

    知道问题产生的原因,就可以对症下药了。

    方案1:  修改mysql的wait_timeout值为一个非常大的值。

                 此方法不太可取,可能会产生大量的sleep连接,导致mysql连接上限了, 建议不使用。

    方案2:每次query之前主动进行连接检测

                 如果是用mysqli,可用内置的mysqli_ping

                 示例:

      if (!$mysqli->ping()) {  

         mysqli->connect(); //重连

    }

                 如果是pdo,可以检测mysql server的服务器信息来判断:

            try {

                    $pdo->getAttribute(\PDO::ATTR_SERVER_INFO);

                } catch (\Exception $e) {

                    if ($e->getCode() == 'HY000') {

                        $pdo = new PDO(xxx);  //重连

                    } else {

                        throw $e;

                    }

                }

 

但这个方案有个缺点:额外多一次请求,所以改进方法:  用一个全局变量存放最后一次query的时间,下一次query的时候先和现在时间对比一下,超过waite_timeout再重连. 或者也可以用swoole_tick定时检测。

          方案3:被动检测, 每次query用try catch包起来,如有mysql gone away异常,则重新连接,再执行一次当前sql.


                    示例:

               try {

                    query($sql);

                } catch (\Exception $e) {

                    if ($e->getCode() == 'HY000') {

                        reconnect(); //重连

                        query($sql)

 

                    } else {

                        throw $e;

                    }

                }

 

        方案4: 用短连接,务必每次操作完之后,手动close

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值