Yii2 解决2006 MySQL server has gone away问题
Yii2 解决2006 MySQL server has gone away问题
Yii2版本 2.0.15.1
php后台任务经常包含多段sql,如果php脚本执行时间较长,或者sql执行时间较长,经常会碰到mysql断连,报2006 MySQL server has gone away
错误。通常,mysql断连了,重连数据库就好了,但是在哪里执行重连呢?这是一个值得思考的问题。
手动重连
最直接的解决办法,是在执行较长sql,或者脚本执行合适的时机,手动重连
\Yii::$app->db->close();
\Yii::$app->db->open();
这里有几个问题
- sql执行时间不好判断,容易受数据库压力的影响。
- 插入重连代码的时机不好判断,太频繁的重连会影响性能。
- 尽管已经充分考虑到插入重连数据库代码的位置,但是依然有"失手"的可能,不能保证完全解决问题。
- 每个数据库都需要充分考虑,例如
\Yii::$app->db1->close()
,代码可复用性不高。
需要时重连
捕获mysql断连异常,在异常处理中重连数据库,重新执行sql。
通常,使用php原生的PDO
类连接数据库的操作步骤是
// 1. 连接数据库
$pdo = new PDO();
// 2. 执行prepare
$stm = $pdo->prepare(sql);
// 3. 绑定参数
$stm->bindValue();
// 4. 执行
$stm->query();
$stm->exec();
在Yii2框架中执行sql,通常有两种方式
- 使用
ActiveRecord
$user = new app\models\User();
$user->name = 'name';
$user->update();
- 拼sql
// 查询类sql select
$sql = <<<EOL
select * from user where name = ':name' limit 1;
EOL;
\Yii::$app->db->createCommand($sql, [':name' => 'name'])->queryAll();
// 更新类sql insert, update, delete...
$sql = <<<EOL
update xm_user set name = 'name1' where name = ':name';
EOL;
\Yii::$app->db->createCommand($sql, [':name' => 'name'])->execute();
在Yii2中,sql的执行,都会调用yii\db\Connection
类的createCommand()
方法获得yii\db\Command
实例。由yii\db\Command
类的queryInternal()
方法执行查询类sql,execute()
方法执行更新类sql。
这里的yii\db\Connection
类似于PDO
类,代表数据库连接, yii\db\Command
类类似于PDOStatement
类, 它的$pdoStatement
属性,保存生成的PDOStatement
句柄。
于是我们改写这两个方法,实现捕获mysql断连异常,重连数据库。
use yii\db\Command;
class MysqlCommand extends Command
{
public function __construct($config = [])
{
parent::__construct($config);
}
protected function queryInternal($method, $fetchMode = null)
{
try {
return parent::queryInternal($method, $fetchMode);
} catch (\yii\db\Exception $e) {
if ($e->errorInfo[1] == 2006 || $e->errorInfo[1] == 2013) {
echo '重连数据库';
$this->db->close();
$this->db->open();
$this->pdoStatement = null;
return parent::queryInternal($method, $fetchMode);
}
throw $e;
}
}
public function execute()
{
try {
return parent::execute();
} catch (\yii\db\Exception $e) {
if ($e->errorInfo[1] == 2006 || $e->errorInfo[1] == 2013) {
echo '重连数据库';
$this->db->close();
$this->db->open();
$this->pdoStatement = null;
return parent::execute();
}
throw $e;
}
}
}
$this->pdoStatement = null
是必要的,否则即使重连了数据库,这里再次执行queryInternal()
或execute()
时,仍会使用原来生成的PDOStatement
句柄,还是会报错。
yii\db\Exception
是Yii实现的Mysql异常,帮我们解析了Mysql抛出的异常码和异常信息, 2006
和2013
均是Mysql断连异常码。
捕获到mysql异常后执行$this->db->close()
,这里的$db
是使用createCommand()
方法传入的db实例, 所以我们也无需要判断db实例是哪一个。
如何使得在调用createCommand()
方法的时候,生成的使我们重写的子类MysqlCommand
而不是默认的yii\db\Command
呢?
阅读代码
public function createCommand($sql = null, $params = [])
{
$driver = $this->getDriverName();
$config = ['class' => 'yii\db\Command'];
if ($this->commandClass !== $config['class']) {
$config['class'] = $this->commandClass; // commandClass属性能覆盖默认的yii\db\Command类
} elseif (isset($this->commandMap[$driver])) {
$config = !is_array($this->commandMap[$driver]) ? ['class' => $this->commandMap[$driver]] : $this->commandMap[$driver];
}
$config['db'] = $this;
$config['sql'] = $sql;
/** @var Command $command */
$command = Yii::createObject($config);
return $command->bindValues($params);
}
我们发现,只要修改yii\db\Connection
的commmandClass
属性就能修改创建的Command
类。
在db.php
配置中加上
'db' => [
'class' => 'yii\db\Connection',
'commandClass' => 'path\to\MysqlCommand', // 加上这一条配置
'dsn' => '',
'username' => '',
'password' => '',
'charset' => 'utf8',
],
这样的配置,要保证使用Yii2提供的\Yii::createObject()
方法创建对象才能生效。
做完以上的修改,在执行拼sql类的查询且不绑定参数时没有问题,但是在使用ActiveRecord
类的方法或者有参数绑定时会报错
SQLSTATE[HY093]: Invalid parameter number: no parameters were bound
说明我们的sql没有绑定参数。
为什么会出现这个问题?
仔细阅读yii\db\Command
的queryInternal()
和execute()
方法,发现他们都需要执行prepare()
方法获取PDOStatement
实例, 调用bindPendingParams()
方法绑定参数。
public function prepare($forRead = null)
{
if ($this->pdoStatement) {
$this->bindPendingParams(); // 绑定参数
return;
}
$sql = $this->getSql();
if ($this->db->getTransaction()) {
// master is in a transaction. use the same connection.
$forRead = false;
}
if ($forRead || $forRead === null && $this->db->getSchema()->isReadQuery($sql)) {
$pdo = $this->db->getSlavePdo();
} else {
$pdo = $this->db->getMasterPdo();
}
try {
$this->pdoStatement = $pdo->prepare($sql);
$this->bindPendingParams(); // 绑定参数
} catch (\Exception $e) {
$message = $e->getMessage() . "\nFailed to prepare SQL: $sql";
$errorInfo = $e instanceof \PDOException ? $e->errorInfo : null;
throw new Exception($message, $errorInfo, (int) $e->getCode(), $e);
}
}
protected function bindPendingParams()
{
foreach ($this->_pendingParams as $name => $value) {
$this->pdoStatement->bindValue($name, $value[0], $value[1]);
}
$this->_pendingParams = []; // 调用一次之后就被置空了
}
这里的$this->_pendingParams
是在调用createCommand()
方法时传入的。
但是调用一次之后,执行了$this->_pendingParams = []
将改属性置空,所以当我们重连数据库之后,再执行到绑定参数这一步时,参数为空,所以报错。
本着软件开发的"开闭原则",对扩展开发,对修改关闭,我们应该重写一个子类,修改掉这个方法,但是这个方法是private
的,所以只能注释掉该语句了。
总结
- 重写
yii\db\Command
类的queryInternal()
和execute()
方法,捕获mysql断连异常。 - 在
db.php
中增加commandClass
配置,使得生成的Command
类为我们重写的子类。 - 注释掉
yii\db\Connection
中bindPendingParams()
方法的$this->_pendingParams = []
语句,保证重新执行时可以再次绑定参数。
Yii2 解决2006 MySQL server has gone away问题的更多相关文章
- MySQL(Navicat)运行.sql文件时报错:[Err] 2006 - MySQL server has gone away 的解决方法
背景: 今天导入一个数据量很大的.sql文件时,报错: 原因: 可能是sql语句过长,超过mysql通信缓存区最大长度. 解决:1. 编辑 MySQL 安装目录下的 my.ini,在最后添加以下内容: ...
- flask+mako+peewee(下)(解决了Error 2006: MySQL server has gone away)
这篇主要介绍在这次项目中使用的peewee 文档地址:http://peewee.readthedocs.org/en/latest/index.html 首先我们要初始化一个数据库连接对象.这里我使 ...
- ThinkPHP出现General error: 2006 MySQL server has gone away的解决方法
错误: #13 {main}SQLSTATE[HY000]: General error: 2006 MySQL server has gone awayFILE: \ThinkPHP\Library ...
- Django (2006, 'MySQL server has gone away') 本地重现与解决
最近我们的Django项目供Java Sofa应用进行tr调用时, 经常会出现一个异常: django.db.utils.OperationalError: (2006, 'MySQL server ...
- SQLyog恢复数据库报错解决方法【Error Code: 2006 - MySQL server has gone away】
https://blog.csdn.net/niqinwen/article/details/8693044 导入数据库的时候 SQLyog 报错了 Error Code: 2006 – MySQL ...
- #2006 - MySQL server has gone away 问题解决方法 (全) (转)
#2006 - MySQL server has gone away 问题解决方法 原文地址:http://www.cnblogs.com/bisonjob/archive/2009/08/18/15 ...
- MYSQL导入数据报错|MYSQL导入超大文件报错|MYSQL导入大数据库报错:2006 - MySQL server has gone away
导SQL数据库结构+数据时,如果数据是批量插入的话会报错:2006 - MySQL server has gone away. 解决办法:找到你的mysql目录下的my.ini配置文件(如果安装目录没 ...
- 2006 - MySQL server has gone away
mysql出现ERROR : (2006, 'MySQL server has gone away') 的问题意思就是指client和MySQL server之间的链接断开了. 造成这样的原因一般是s ...
- mysql 导入数据是报错:2006 - MySQL server has gone away
导SQL数据库结构+数据时,如果数据是批量插入的话会报错:2006 - MySQL server has gone away. 解决办法:找到你的mysql目录下的my.ini配置文件,加入以下代码 ...
随机推荐
- vue2.0填坑有感(持续更新ing)
1.请求数据 用本地json数据进行mock的时候,一般放在created 过程就Ok了,这样可以尽早获取数据:如果有依赖dom必须存在的清空,就放到mounted里面,具体用法如下所示: // cr ...
- 玩转Web之SSH--Heibernate (一)---第一个demo
最近在学heibernate,是看马士兵老师的视频学的,在这里总结一下,做点笔记.关于heibernate的优点,大家可以在网上 百度,这里不做赘述,直接讲怎么使用heibernate 步骤一:新建项 ...
- 12.app后端如何选择合适的数据库产品
app后端的开发中,经常要面临的一个问题是:数据放在哪里? mysql ?redis?mongodb? 现在有这么多优秀的开源数据库产品,怎么根据业务场景来选择合适的数据? 常用的数据库产品的优缺点又 ...
- python二维码生成器
周小董简书主页二维码.png 周小董博客主页二维码.png 现在,我们生活中到处可以看到二维码.它有啥好处呢?它具有信息容量大.可靠性高.可表示汉字及图象多种文字信息.保密防伪性强等优点. 我们生 ...
- dictionary.go
package sego import "github.com/adamzy/cedar-go" // Dictionary结构体实现了一个字串前缀树,一个分词可能出现在叶子节点也 ...
- Java7里try-with-resources分析
这个所谓的try-with-resources,是个语法糖.实际上就是自动调用资源的close()函数.和Python里的with语句差不多. 例如: [java] view plain copy ...
- 【贪心+背包】BZOJ1334 [Baltic2008]Elect
Description 从N个数中选出任意个数且和尽量大,但要满足去掉任意一个和就小于总和的一半.n<=300, ai<=1e5. Solution 这个条件其实就是 去掉选出的最小的一个 ...
- iOS 社交化分享功能
iOS 开发过程中可能会遇到需要进行第三方分享的需求,比如向QQ,微信,微博等分享 如下图 我们今天要讲到的方式是使用了一个第三方工具: http://www.sharesdk.cn 一,注册账号 去 ...
- Postman----设置代理抓取手机上的请求
一般为方便接口测试,我们都会设置代理,抓取手机上的请求来方便测试,具体的操作方法如下. 操作步骤: 一.手机和电脑连接同一网络,查看电脑连接网络的IP,配置手机的代理 1.查看电脑连接的网络与IP地址 ...
- C# WinForm ShowInTaskbar Api Version
using System; using System.Runtime.InteropServices; namespace x { unsafe class NativeWindow { /* * W ...