很多开发人员都会遇见”MySQL: ERROR 1040: Too many connections”的异常情况,造成这种情况的一种原因是访问量过高,MySQL服务器抗不住,这个时候就要考虑增加从服务器分散读压力;另一种原因就是MySQL配置文件中max_connections值过小。

查看mysql的最大连接数:

mysql> show variables like '%max_connections%';



查看服务器响应的最大连接数:

mysql> show global status like 'Max_used_connections';



可以看到服务器响应的最大连接数为2,远远低于mysql服务器允许的最大连接数值。



对于mysql服务器最大连接数值的设置范围比较理想的是:服务器响应的最大连接数值占服务器上限连接数值的比例值在10%以上,如果在10%以下,说明mysql服务器最大连接上限值设置过高。



Max_used_connections / max_connections * 100% = 2/151 *100% ≈ 1%

我们可以看到占比远低于10%(因为这是本地测试服务器,结果值没有太大的参考意义,大家可以根据实际情况设置连接数的上限值)。



设置这个最大连接数值:


mysql> set GLOBAL max_connections=256;

连接数超过了 MySQL 设置的值,与 max_connections 和 wait_timeout 都有关系。wait_timeout 的值越大,连接的空闲等待就越长,这样就会造成当前连接数越大。


wait_timeout过大有弊端 默认值为28800(即8小时),其体现就是MySQL里大量的SLEEP进程无法及时释放,拖累系统性能,不过也不能把这个指设置的过小,否则你可能会遭遇到“MySQL has gone away”之类的问题,通常来说,我觉得把wait_timeout设置为10是个不错的选择,但某些情况下可能也会出问题,比如说有一个CRON脚本,其中两次SQL查询的间隔时间大于10秒的话,那么这个设置就有问题了(当然,这也不是不能解决的问题,你可以在程序里时不时mysql_ping一下,以便服务器知道你还活着,重新计算wait_timeout时间)

mysql> show global variables like 'wait_timeout';
mysql> set global wait_timeout=10;


生产服务器上数据库又不能随便重启,只好想办法手动去释放一些没用的连接。

mysql> show processlist;可以得到所有连接到这个服务器上的MySQL连接

你可以看到MySQL数据连接列表,而且每一个都会有一个进程ID号(在上表的第一列)。我们只要输入这样的命令:
mysql> kill 1180421;(其中1180421是进程列表里找到并且要杀掉的进程号)



为了防止发生too many connections时候无法登录的问题,mysql manual有如下的说明:


mysqld actually allows max_connections+1 clients to connect. The extra connection is reserved for use by accounts that have the SUPER privilege. By granting the SUPER privilege to administrators and not to normal users (who should not need it), an administrator can connect to the server and use SHOW PROCESSLIST to diagnose problems even if the maximum number of unprivileged clients are connected.



因此, 必须只

赋予root用户的SUPER权限,同时所有数据库连接的帐户不能赋予SUPER权限。前面说到的报错后无法登录就是由于我们的应用程序直接配置的root用户


查看当前数据库的所有用户:
 select user,host,password from mysql.user;

 给用户赋予super权限(super和ALL PRIVILEGES都可以):
 GRANT super ON *.* TO 'mysql'@'localhost';
 GRANT ALL PRIVILEGES ON *.* TO 'mysql'@'localhost';

 删除用户的super权限(super和ALL PRIVILEGES都可以):
 REVOKE super ON *.* FROM 'mysql'@'localhost';
 REVOKE ALL PRIVILEGES ON *.* FROM 'mysql'@'localhost';

 查看赋予用户的权限
 SHOW GRANTS FOR 'mysql'@'localhost';