连接数从异常到 300 到 5(RDS MySQL 的一个大坑•后记)
在 《记 RDS MySQL 的一个大坑》 中,我提到遇到 User juxxxxxxxxxx already has more than 'max_user_connections' active connections…… 这样的错误,最终通过在循环中使用 Thread.Sleep,降低 CRUD 操作的频率,让连接数下降至不到原来的一半,从而解决了这个棘手的问题,有兴趣的朋友可以点击链接回顾一下。
今天又看了一下添加 Thread.Sleep 后,程序运行时的 IOPS 和 连接数:

运行结果:连接数:300,运行时间:68 分钟,IOPS:7。
昨天在博客园中发出上篇文章后,热心的朋友(@沈赟,@不知道风往哪儿吹)对此问题提出了宝贵的意见和想法,激发了我对此问题继续深究的决定。
下午经过几个小时的分析和测试,终于找到了该问题的真正原因和更好的解决方法,在此做个补充。
真正的原因在于:使用 MySQL 官方提供的 MySql.Data 作为驱动程序连接 MySQL 数据库的时候,默认使用了连接池,才引发了这个问题。错怪了阿里云(上篇中提到怀疑阿里云改了 MySQL 底层做了限制),在此对自己的不严谨表示诚恳的道歉,凡事要自己多思考多研究多求证,不可轻易怀疑权威的力量,切记切记!!!
下面聊一聊该问题出现的真正原因和更优的解决方法。
MySQL 连接池
根据官方介绍:MySQL Connector/NET 中(即 MySql.Data 中)连接池的工作的机制是,当客户端配置 MySqlConnection 时,连接池通过保持一组与服务器的本地连接使其处于活动状态,随后,如果打开一个新的 MySqlConnection 对象,它将从连接池中创建连接,而不是重新创建一个新的本地连接。这样便可以重用数据库连接,避免了频繁创建、释放连接引起的大量性能开销,这有助于缩短响应时间、统一管理、提高运行性能等等。
在软件开发中,大多数情况下,数据库连接都有重用的可能,即便永不重用,连接池也有自己的回收机制在适当的时候释放资源,这有点像带有过期时间的缓存数据,也像 .NET 的 GC 回收机制。正因为在大多数情况下,它可以提高运行的性能,也有完善且可配置的回收机制。所以在没有提供任何连接池选项的情况下,MySQL Connector/NET 默认启用连接池,也就是说,创建 MySqlConnection 时使用下面的连接字符串:
server=xxx;port=3306;userid=myuserid;password=pwd123;database=db125;charset=utf8;
等同于使用:
server=xxx;port=3306;userid=myuserid;password=pwd123;database=db125;charset=utf8;Pooling=true;
而我原来的代码中恰好用的就是前者,也就是说默认启用了连接池。
为什么使用连接池反而出问题了
上面说到连接池有那么多的好处,为什么我用了连接池反倒出问题了呢?我们来看一下
连接对池资源的利用情况:
官方文档:
Connector/NET runs a background job every three minutes and removes connections from pool that have been idle (unused) for more than three minutes. The pool cleanup frees resources on both client and server side. This is because on the client side every connection uses a socket, and on the server side every connection uses a socket and a thread.
译文:
Connector/NET 每三分钟运行一次后台作业,从连接池中删除闲置(未使用)超过三分钟的连接。连接池清理会释放客户端和服务器端的资源。这是因为在客户端,每个连接使用一个套接字,而在服务器端,每个连接都使用一个套接字和一个线程。
上一篇中有介绍过我的程序的基本情况,这里有必要再补充一下关键的使用场景:
我们的 MySql 服务实例有很多台,每台实例上有很多个数据库,只有其中一台 MySql 服务实例出现了超出 max_user_connections 的异常,这台实例最大的连接数限制在 600,但是这台实例上的数据库就有 700 多个。
聪明的朋友看到这里,估计已经明白为什么使用了连接池会出现问题了。为什么呢?就因为上面提到的连接池每三分钟运行一次清理操作呗。循环语句执行的速度是很快的,有的小库瞬间就执行完了,但是在连接池中却保持了一个连接,还没有到每隔三分钟的资源回收时间(这也是我在上篇中添加了 Thread.Sleep 后连接数减少的原因)。当这台实例的 600 个连接被全部占满时,再连接同一实例上另一个连接池中没有缓存的数据库时,就报了超出 max_user_connections 的异常。
解决方法
怎么解决呢?最简单的解决方法就是,判断请求的是这台 MySql 服务实例时,不使用连接池,这样就会在调用 MySqlConnection 的 Close 方法时,立即释放客户端和服务端所占用的资源。因此,在数据库连接字符串中加上 Pooling=false,改成下面这样:
server=xxx;port=3306;userid=myuserid;password=pwd123;database=db125;charset=utf8;Pooling=false;
然后发布到服务器上进行测试,查看一下程序运行时的 IOPS 和 连接数:

惊不惊喜,意不意外,嚯嚯嚯~~~
连接数从异常到 300 到 5(RDS MySQL 的一个大坑•后记)的更多相关文章
- 记阿里云 RDS MySQL 的一个大坑
花了一个下午的时间,终于把一个阿里云 RDS MySQL 的一个大坑填上了,解决方法令人匪夷所思!绝对会让各位看官感到大吃一惊,阿里云 RDS MySQL 居然有这样 xx 的大坑! 问题 最近应业务 ...
- RDS MySQL 连接数满情况的处理
RDS MySQL 连接数满情况的处理 RDS MySQL 连接数满有2种情况 1. 空闲连接过多 原因: 应用使用长连接模式 - 对于长连接模式(比如Java应用),应用侧应该配置连接池.连接池的初 ...
- 实测:云RDS MySQL性能是自建的1.6倍
1. 摘要 基于之前写的「云厂商 RDS MySQL 怎么选」的文章,为了进一步了解各云厂商在RDS MySQL数据库性能上的差异,本文将对自建MySQL.阿里云.腾讯云.华为云和AWS 的 RDS ...
- RDS MySQL InnoDB 锁等待和锁等待超时的处理
https://help.aliyun.com/knowledge_detail/41705.html 1. Innodb 引擎表行锁等待和等待超时发生的场景 2.Innodb 引擎行锁等待情况的处理 ...
- RDS MySQL 空间问题的原因和解决
来源:https://help.aliyun.com/knowledge_detail/41739.html RDS MySQL 空间问题的原因和解决 更新时间:2016-07-22 17:20:14 ...
- Configure Amazon RDS mysql to store Chinese Characters
Configure Amazon RDS mysql to store Chinese Characters https://dev.mysql.com/doc/refman/5.7/en/chars ...
- RDS MySQL 全文检索相关问题的处理
RDS MySQL 全文检索相关问题 1. RDS MySQL 对全文检索的支持 2. RDS MySQL 全文检索相关参数 3. RDS MySQL 全文检索中文支持 3.1 MyISAM 引擎表 ...
- RDS——mysql主从
测试一:RDS主 RDS从 只需要创建一个RDS数据库实例 1.版本 RDS:5.6.21 2.操作步骤 选中被操作的数据--->点击实例操作---->创建只读副本---& ...
- 更改 AWS RDS mysql时区 -摘自网络
AWS RDS AWS上搭建数据库的时候,不是DB on EC2就是RDS,但是选择RDS时,Timezone怎么处理? 「面向全球提供的AWS来讲理所当然的是UTC」,而RDS也不是例外.把服务器迁 ...
随机推荐
- RocketMQ4.x安装部署
1.下载安装包:https://www.apache.org/dyn/closer.cgi?path=rocketmq/4.2.0/rocketmq-all-4.2.0-bin-release.zip ...
- tcpack--4延时ack
TCP在收到数据后必须发送ACK给对端,但如果每收到一个包就给一个ACK的话会使得网络中被注入过多报文.TCP的做法是在收到数据时不立即发送ACK,而是设置一个定时器,如果在定时器超时之前有数据发送给 ...
- tcp syn-synack-ack 服务端接收ack
TCP 服务端 接收到ack tcp_v4_rcv() -> tcp_v4_do_rcv() -> tcp_v4_hnd_req() + tcp_child_process()tcp_v4 ...
- 查找数组中第k大的数
问题: 查找出一给定数组中第k大的数.例如[3,2,7,1,8,9,6,5,4],第1大的数是9,第2大的数是8-- 思考:1. 直接从大到小排序,排好序后,第k大的数就是arr[k-1]. 2. ...
- 自行实现的jar包中,日志库的适配实现
日常情况下,我们自己都会自行实现一些基础的jar包,如dao包.service包或一些其他完成特定功能的jar包.如果没有一套调试日志信息,出现问题时想查找问题非常不方便.可能大多数小伙伴都会有自 ...
- C++深拷贝与浅拷贝区别
浅拷贝只是对指针的拷贝,浅拷贝后两个指针指向同一个内存空间: 深拷贝不仅对指针进行拷贝,而且对指针指向的内容进行拷贝,经深拷贝后的指针是指向两个不同地址的指针. 当对一个已知对象进行拷贝时,编译系统会 ...
- Ubuntu16.04安装搜狗输入法报错:dkpg:处理归档sogoupinyin.deb(--install)时出错,安装sogoupinyin将破坏fcitx-ui-qimpanel
系统:ubuntu16.04 事件:安装搜狗拼音时报错 报错信息(ubuntu语言是英文的报错信息): dpkg: regarding sogoupinyin_2.3.2.07_amd64-831.d ...
- Vegas的软对比具体如何设置
软对比是Vegas中比较常用的一个视频特效,通过这一特效可以让视频画面更加柔和.而对于新手用户来说,软对比显得有些陌生,不知道该如何使用. 本集主要为大家介绍:sony vegas的软对比. 在视频制 ...
- MyBatis的一二级缓存
一级缓存 一级缓存默认是开启的,生命周期和SqlSession相同.一个会话中每次执行一个查询操作时,会先查询二级缓存,如果二级缓存没查到或者二级缓存未开启就会从一级缓存中查询,如果一级缓存也未查到就 ...
- 【xmind converse excel】测试用例定制化小工具
背景 公司使用jira, jira写测试用例,jira可以通过execl导入进jira, 生成测试用例,但是模板很不统一,如果只是再execl中修改,又觉得及其的麻烦,所以写了一个xmind 转化为定 ...