# 如何解决Mysql异常:Communications link failure
# 问题现象
MySQL数据加工或者计算报表时偶尔会报下列异常。
异常1:
java.io.EOFException: Can not read response from server. Expected to read 4 bytes, read 0 bytes before connection was unexpectedly lost.
at com.mysql.cj.protocol.FullReadInputStream.readFully(FullReadInputStream.java:67)
at com.mysql.cj.protocol.a.SimplePacketReader.readHeader(SimplePacketReader.java:63)
at com.mysql.cj.protocol.a.SimplePacketReader.readHeader(SimplePacketReader.java:45)
at com.mysql.cj.protocol.a.NativeProtocol.readMessage(NativeProtocol.java:556)
[wrapped] com.mysql.cj.exceptions.CJCommunicationsException: Communications link failure
The last packet sent successfully to the server was 0 milliseconds ago. The driver has not received any packets from the server.
at jdk.internal.reflect.GeneratedConstructorAccessor117.newInstance(Unknown Source)
at java.base/jdk.internal.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)
at java.base/java.lang.reflect.Constructor.newInstance(Constructor.java:490)
at com.mysql.cj.exceptions.ExceptionFactory.createException(ExceptionFactory.java:61)
at com.mysql.cj.exceptions.ExceptionFactory.createException(ExceptionFactory.java:105)
at com.mysql.cj.exceptions.ExceptionFactory.createException(ExceptionFactory.java:151)
at com.mysql.cj.exceptions.ExceptionFactory.createCommunicationsException(ExceptionFactory.java:167)
at com.mysql.cj.protocol.a.NativeProtocol.readMessage(NativeProtocol.java:562)
at com.mysql.cj.protocol.a.NativeProtocol.readServerCapabilities(NativeProtocol.java:514)
at com.mysql.cj.protocol.a.NativeProtocol.beforeHandshake(NativeProtocol.java:404)
at com.mysql.cj.protocol.a.NativeProtocol.connect(NativeProtocol.java:1447)
异常2:
10:45:05.709
正在执行导入,已写入行数:0
10:45:15.861
LOAD DATA LOCAL INFILE 'x' ......
10:57:43.397
正在执行导入,已写入行数:1024000
11:00:28.874
Communications link failure
The last packet successfully received from the server was 913,010 milliseconds ago. The last packet sent successfully to the server was 165,478 milliseconds ago.异常堆栈:
java.net.SocketException: 断开的管道 (Write failed)
at java.base/java.net.SocketOutputStream.socketWrite0(Native Method)
at java.base/java.net.SocketOutputStream.socketWrite(SocketOutputStream.java:110)
at java.base/java.net.SocketOutputStream.write(SocketOutputStream.java:150)
at java.base.security.ssl.SSLSocketOutputRecord.deliver(SSLSocketOutputRecord.java:341)
at java.base.security.ssl.SSLSocketImpl$AppOutputStream.write(SSLSocketImpl.java:1186)
at java.base/java.io.BufferedOutputStream.write(BufferedOutputStream.java:123)
# 问题原因
MySQL数据库服务重启了,而应用的数据库连接池缓存还是旧的数据库连接。
net_read_timeout
、net_write_timeout
参数默认配置的时间过短,一般发生在数据加工数据迁移场景下。以两个MySQL使用数据加工迁移数据为例说明,数据加工抽取数据采用MySQL load data 加管道流的方式实现,由于源头获取数据被卡住了,在往目标库写入数据时超过MySQLnet_read_timeout
参数的限制,该参数默认为30秒,即30秒MySQL没有从jdbc连接中获取数据,就会自动断掉连接。上述异常说明如下:- The last packet sent successfully to the server was 165,478 milliseconds ago ,表示在165秒以前成功传输过数据,往MySQL写入数据连接是30秒,而错误是在165秒发生,是因为在这165秒之间没有写入数据了,在165秒以后待管道的源表在次获取数据后进行写入时,这时连接超时抛出了异常。
- The last packet successfully received from the server was 913,010 milliseconds ago. 表示建立连接执行load data 后,MySQL给jdbc客户端的响应;从上面的异常日志看出,10:45:15开始导入,异常11:00:28抛出,预计913秒左右。
注意:
MySQL默认配置了 wait_timeout=28800
,8个小时连接没有使用会自动回收掉,回收后连接池不知道继续使用被回收的连接也会导致上述异常,但在SuccBI中默认每半小时检查一下连接的有效性,MySQL数据库自动回收空闲连接应该不存在。
# 排查步骤
连接池有效性检查
配置连接池的有效性,参考数据源管理:数据源管理。如果是配置默认连接池,修改工作目录下的jdbc.conf中的testConnectionOnCheckout
参数为true,参考如下:
{
"driver": "com.mysql.cj.jdbc.Driver",
"dbType": "MySQL",
"url": "jdbc:mysql://xxxxxxx:3361/whhsjc?useUnicode=true&characterEncoding=utf8&allowLoadLocalInfile=true",
"user": "xxxxx",
"password": "xxxxx",
"testConnectionOnCheckout": "true"
}
MySQL timeout参数检查
查看当前MySQL数据库的超时参数,在数据库中执行
show global variables like '%timeout%'
VARIABLE_NAME VALUE connect_timeout 10 delayed_insert_timeout 300 have_statement_timeout YES innodb_flush_log_at_timeout 1 innodb_lock_wait_timeout 2 innodb_rollback_on_timeout OFF interactive_timeout 28800 lock_wait_timeout 31536000 mysqlx_connect_timeout 30 mysqlx_idle_worker_thread_timeout 60 mysqlx_interactive_timeout 28800 mysqlx_port_open_timeout 0 mysqlx_read_timeout 30 mysqlx_wait_timeout 28800 mysqlx_write_timeout 60 net_read_timeout 30 net_write_timeout 60 rpl_stop_slave_timeout 31536000 slave_net_timeout 60 wait_timeout 28800
在
my.cnf
(windows为my.ini)中修改net_read_timeout
、net_write_timeout
参数值,修改Mysql配置文件参考Mysql安装配置net_read_timeout=1800 net_write_timeout=1800
上述参数单位为秒,修改后需要重启MySQL服务才能生效。
说明:
net_read_timeout
表示MySQL服务器从客户端读取数据的网络超时,即客户端向服务器提交数据时,等待多少秒仍未执行成功时自动断开连接net_write_timeout
表示MySQL服务器往客户端写数据的网络超时,即表示客户端从服务器端读取数据,等待多少秒仍未执行成功时自动断开连接
0条评论
评论