长连接和短连接
概念解释:
长连接:
长时间保持客户端与服务端的连接状态。默认超时时间8小时
短连接:
1 | 数据传输完毕立即断开,每次连接只完成一项业务的发送。 |
短连接的原理:
客户端连接--创建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个连接
1 2 3 | 但是每个连接几乎不到1s钟就关闭了,很典型的频繁连接进行数据通信。这种业务就特别的消耗系统资源和mysql的资源。这种情况业务应该使用长连接来使用数据库服务。 更改为长连接后,审计连接用户数就马上正常了,保持在一定的数值。 |
个人总结:
长连接主要用于在少数客户端与服务端的频繁通信,因为这时候如果用短连接频繁通信常会发生Socket出错,并且频繁创建Socket连接也是对资源的浪费。
但是对于服务端来说,长连接也会耗费一定的资源,需要专门的线程(unix下可以用进程管理)来负责维护连接状态
1 2 | 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个小时单位是秒):
1 2 3 4 5 6 7 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.\n", my_dbname, my_hostname);
use_mysql = 0;
} else {
char value = 1;
mysql_options(&logdb, MYSQL_OPT_RECONNECT, (char*)&value);
use_mysql = 1;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 25岁的心里话
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现