概念解释:
长连接:
长时间保持客户端与服务端的连接状态。默认超时时间8小时
短连接:
数据传输完毕立即断开,每次连接只完成一项业务的发送。
短连接的原理:
客户端连接--创建socket认证连接--维护连接--数据传输--关闭连接 #连接-》数据传输-》关闭连接;
长连接的原理:
客户端连接--创建socket认证连接--维护连接--数据传输--维护连接--数据传输.....-关闭连接 #连接-》数据传输-》保持连接-》数据传输-》保持连接-》…………-》关闭连接;
长短连接的例子
该使用长连接的情况下使用了短连接
某业务在审计时候发现连接数一直在直线上升:
mysql> select count(1) from db_monitor.accesslog; +----------+ | count(1) | +----------+ | 16117 | +----------+ 1 row in set (0.01 sec) mysql> select count(1) from db_monitor.accesslog; +----------+ | count(1) | +----------+ | 23768| +----------+ 1 row in set (0.01 sec) ...
截一段连接的审计图给看看:
查看mysql的进程却只有2-3个连接
但是每个连接几乎不到1s钟就关闭了,很典型的频繁连接进行数据通信。这种业务就特别的消耗系统资源和mysql的资源。这种情况业务应该使用长连接来使用数据库服务。 更改为长连接后,审计连接用户数就马上正常了,保持在一定的数值。
个人总结:
长连接主要用于在少数客户端与服务端的频繁通信,因为这时候如果用短连接频繁通信常会发生Socket出错,并且频繁创建Socket连接也是对资源的浪费。
但是对于服务端来说,长连接也会耗费一定的资源,需要专门的线程(unix下可以用进程管理)来负责维护连接状态
1、在频繁的与数据库服务通信,并且又非高并发的情况下,使用长连接更合适; 2、太多持久连接,大部分是sleep状态的,或者系统是高并发的,使用短连接更合适。
+++++++++++++++++++++
首先,如果使用了长连接在闲置时间超过timeout值后,mysql server就会关闭此连接,而客户端在执行查询的时候就会得到一个类似于“MySQL server has gone away“这样的错误。
在 使用mysql_real_connect连接数据库之后,再使用mysql_options( &mysql, MYSQL_OPT_RECONNECT, … ) 来设置为自动重连。这样当mysql连接丢失的时候,使用mysql_ping能够自动重连数据库。如果是在mysql 5.1.6之前,那么则应在每次执行完real_connect 之后执行mysql_options( &mysql, MYSQL_OPT_RECONNECT, … ) ,如果是mysql 5.1.6+,则在connect之前执行一次就够了。
查看mysql连接数
mysqladmin -uroot -p processlist
当设置了MYSQL_OPT_RECONNECT为1时,超时后再查看processlist,则自动建立的连接不在列表中,但事实上连接确实建立并被使用了。
在MYSQL的默认设置中,长连接超时断开后,后续在该连接上的查询操作都将失败。
解决办法一:修改MYSQL服务器的配置参数
长连接的超时时间默认8小时,/etc/my.cnf中添加一行wait_timeout=超时时间 。
临时配置方法:
用超级用户登录MYSQL,注意必须是超级用户。然后输入
目前默认的超时时间(8个小时单位是秒):
show global variables like 'wait_timeout'; +---------------+-------+ | Variable_name | Value | +---------------+-------+ | wait_timeout | 28800 | +---------------+-------+ 1 row in set (0.00 sec)
将超时时间设置成10个小时,可以输入:
set global wait_timeout=36000; Query OK, 0 rows affected (0.00 sec)
这种方法设置的参数立即生效。但如果/etc/my.cnf中没有配置,则重启服务后,global变量会从/etc/my.cnf中读取新的变量值。
下边是一段示例代码:
if(!mysql_real_connect(&logdb, my_hostname, my_user, my_password, my_dbname, my_port, my_sock, 0)){
ast_log(LOG_ERROR, "Failed to connect to mysql database %s on %s. ", my_dbname, my_hostname);
use_mysql = 0;
} else {
char value = 1;
mysql_options(&logdb, MYSQL_OPT_RECONNECT, (char*)&value);
use_mysql = 1;
}