mysql server has gone away-阿里云开发者社区

开发者社区> 数据库> 正文
登录阅读全文

mysql server has gone away

简介:
mysql server has gone away,他的意思就是说,mysql已经走了。 
为什么会出现这种情况呢?原因主要还是跟两个参数有关: 

interactive_timeout 这个是客户端和服务器端握手的超时时间,如果你执行一个大操作的话,握手的时间就要长一点。太久也不好, 
有时会出现死循环的操作,握手就会一直握下去,直到系统资源耗完。参数默认值:28800秒(8小时) 

wait_timeout 这个是握手完等待的时候,如时超过设置的值,就自动关闭连接进程,所占用的资源释放。参数默认值:28800秒(8小时) 

SHOW VARIABLES LIKE '%timeout%'; 

你可以执行上面的命令,看一下你这两个值的当前的设置 
也可以在my.cnf或者my.ini修改一下。 
interactive_timeout = 120 
wait_timeout = 5 

mysql> show variables like '%timeout%'; 
+-----------------------------+----------+ 
| Variable_name               | Value    | 
+-----------------------------+----------+ 
| connect_timeout             | 10       | 
| delayed_insert_timeout      | 300      | 
| innodb_flush_log_at_timeout | 1        | 
| innodb_lock_wait_timeout    | 50       | 
| innodb_rollback_on_timeout  | OFF      | 
| interactive_timeout         | 28800    | 
| lock_wait_timeout           | 31536000 | 
| net_read_timeout            | 30       | 
| net_write_timeout           | 60       | 
| rpl_stop_slave_timeout      | 31536000 | 
| slave_net_timeout           | 3600     | 
| wait_timeout                | 28800    | 
+-----------------------------+----------+ 
12 rows in set (0.00 sec) 


ySQL server has gone away的原因及解决方案 

常见的原因: 
1. 服务器超时,比如说在一个脚本里连接上服务器后进行了一些数据库操作后,就开始冷落了数据库,执行起其他一些非数据库操作的耗时操作, 
数据库等啊等等啊等,结果超时了,只好关闭连接。如果是这个原因,请加大wait_timeout。 
2. 因为SQL语句太长所致,比如插入一个BLOB列,或者用一条INSERT语句同时插入很多行,因为SQL太长,数据库可能就无法接收到完整的 SQL, 
结果数据库就会觉得客户端出错了,从而关闭连接。如果是这个原因,请加大max_allowed_packet。 


把PHP连接MySQL的连接方式从长连接改成了短连接,就再没出现了。 

1.MySQL服务器的wait_timeout参数为默认的28800秒,即8小时。该参数的意思是MySQL等待睡眠的连接的持续时间,如果超过8小时不活动,MySQL就会关闭这个连接。 
这个参数可以动态修改,但这个参数是session级别的,需要应用重新建立连接获取这个变量值才会生效。 

2.PHP的默认php.ini里面,default_socket_timeout参数是默认60s.default_socket_timeout参数的意思是socket超时为60s。 官方解释 

3.在MySQL的官方文档里面,有专门的章节介绍MySQL Server has gone away。但并没有提及到客户端本身也会有超时机制,以前曾经试过把wait_timeout设置为1年 
,但App还是继续出现MySQL server has gone away的错误。 

4.在PHP程序里面,如果php.ini修改起来不方便,可以以下代码来尝试解决。 

ini_set('mysql.connect_timeout', 300); 

ini_set('default_socket_timeout', 300); 

在ini_set后,可以用ini_get来验证参数设置适合符合预期。 







=================================================================== 
原因1. MySQL 服务宕了 

判断是否属于这个原因的方法很简单,执行以下命令,查看mysql的运行时长 

$ mysql -uroot -p -e "show global status like 'uptime';" 
+---------------+-------+ 
| Variable_name | Value | 
+---------------+-------+ 
| Uptime        | 68928 | 
+---------------+-------+ 
1 row in set (0.04 sec) 

或者查看MySQL的报错日志,看看有没有重启的信息 

$ tail /var/log/mysql/error.log 
130101 22:22:30 InnoDB: Initializing buffer pool, size = 256.0M 
130101 22:22:30 InnoDB: Completed initialization of buffer pool 
130101 22:22:30 InnoDB: highest supported file format is Barracuda. 
130101 22:22:30 InnoDB: 1.1.8 started; log sequence number 63444325509 
130101 22:22:30 [Note] Server hostname (bind-address): '127.0.0.1'; port: 3306 
130101 22:22:30 [Note]   - '127.0.0.1' resolves to '127.0.0.1'; 
130101 22:22:30 [Note] Server socket created on IP: '127.0.0.1'. 
130101 22:22:30 [Note] Event Scheduler: Loaded 0 events 
130101 22:22:30 [Note] /usr/sbin/mysqld: ready for connections. 
Version: '5.5.28-cll'  socket: '/var/lib/mysql/mysql.sock'  port: 3306  MySQL Community Server (GPL) 
如果uptime数值很大,表明mysql服务运行了很久了。说明最近服务没有重启过。 

如果日志没有相关信息,也表名mysql服务最近没有重启过,可以继续检查下面几项内容。 

2. 连接超时 

如果程序使用的是长连接,则这种情况的可能性会比较大。 

即,某个长连接很久没有新的请求发起,达到了server端的timeout,被server强行关闭。 

此后再通过这个connection发起查询的时候,就会报错server has gone away 


$ mysql -uroot -p -e "show global variables like '%timeout';" 
+----------------------------+----------+ 
| Variable_name              | Value    | 
+----------------------------+----------+ 
| connect_timeout            | 30       | 
| delayed_insert_timeout     | 300      | 
| innodb_lock_wait_timeout   | 50       | 
| innodb_rollback_on_timeout | OFF      | 
| interactive_timeout        | 28800    | 
| lock_wait_timeout          | 31536000 | 
| net_read_timeout           | 30       | 
| net_write_timeout          | 60       | 
| slave_net_timeout          | 3600     | 
| wait_timeout               | 28800    | 
+----------------------------+----------+ 
mysql> SET SESSION wait_timeout=5; 

## Wait 10 seconds 

mysql> SELECT NOW(); 
ERROR 2006 (HY000): MySQL server has gone away 
No connection. Trying to reconnect... 
Connection id:    132361 
Current database: *** NONE *** 

+---------------------+ 
| NOW()               | 
+---------------------+ 
| 2013-01-02 11:31:15 | 
+---------------------+ 
1 row in set (0.00 sec) 


3. 进程在server端被主动kill 
这种情况和情况2相似,只是发起者是DBA或者其他job。发现有长时间的慢查询执行kill xxx导致。 

$ mysql -uroot -p -e "show global status like 'com_kill'" 
+---------------+-------+ 
| Variable_name | Value | 
+---------------+-------+ 
| Com_kill      | 0     | 
+---------------+-------+ 

4. Your SQL statement was too large. 
当查询的结果集超过 max_allowed_packet 也会出现这样的报错。定位方法是打出相关报错的语句。 
用select * into outfile 的方式导出到文件,查看文件大小是否超过 max_allowed_packet ,如果超过则需要调整参数,或者优化语句。 

mysql> show global variables like 'max_allowed_packet'; 
+--------------------+---------+ 
| Variable_name      | Value   | 
+--------------------+---------+ 
| max_allowed_packet | 1048576 | 
+--------------------+---------+ 
1 row in set (0.00 sec) 
修改参数: 

mysql> set global max_allowed_packet=1024*1024*16; 
mysql> show global variables like 'max_allowed_packet'; 
+--------------------+----------+ 
| Variable_name      | Value    | 
+--------------------+----------+ 
| max_allowed_packet | 16777216 | 
+--------------------+----------+ 
1 row in set (0.00 sec)

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

分享: