天天看点

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'" 

| 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 | 

修改参数: 

mysql> set global max_allowed_packet=1024*1024*16; 

+--------------------+----------+ 

| variable_name      | value    | 

| max_allowed_packet | 16777216 | 

1 row in set (0.00 sec)