翼度科技»论坛 编程开发 mysql 查看内容

MySQL kill 会话不起作用?

6

主题

6

帖子

18

积分

新手上路

Rank: 1

积分
18

  • GreatSQL社区原创内容未经授权不得随意使用,转载请联系小编并注明来源。
  • GreatSQL是MySQL的国产分支版本,使用上与MySQL一致。
  • 作者: 王权富贵
  • 文章来源:GreatSQL社区原创
背景

在一次日常测试中发现,kill 一个会话后,SQL语句依然在运行并没终止;被kill的会话重新连接并继续执行原来的SQL语句。
测试

本次测试基于MySQL 8.0.27
1.创建测试表
  1. create table t1 (id int, name  varchar(30));   
  2. insert into t1 values (1,'a'),(2,'b');  
复制代码
2.开启3个会话

session1session2session3begin;select * from t1;rename table t1 to t2;   【由于锁等待,hang住】show processlist;   【查看 processlist_id】kill session2;【session2 重新连接并且继续执行语句,处于锁等待状态】show processlist;  【可以看到session2重新连接并继续执行SQL】commit;【rename 执行成功】show tables;  【t1 被 rename 为 t2】session1:开启一个事务不提交
  1. mysql> use test
  2. Database changed
  3. mysql>
  4. mysql>
  5. mysql> begin;
  6. Query OK, 0 rows affected (0.00 sec)
  7. mysql> select * from t1;
  8. +------+------+
  9. | id   | name |
  10. +------+------+
  11. |    1 | a    |
  12. |    2 | b    |
  13. +------+------+
  14. 2 rows in set (0.00 sec)
复制代码

session2:执行DDL语句
  1. mysql> use test
  2. Database changed
  3. mysql>
  4. mysql>
  5. mysql> rename table t1 to t2;
复制代码

session3:kill session2
  1. mysql> show processlist;
  2. +-----+------+---------+---------+--------------------------------+----------------------+
  3. | Id  | db   | Command | Time    | State                          | Info                 |
  4. +-----+------+---------+---------+--------------------------------+----------------------+
  5. |   6 | NULL | Daemon  | 4399013 | Waiting on empty queue         | NULL                 |
  6. | 132 | test | Sleep   |     232 |                                | NULL                 |
  7. | 134 | test | Query   |     123 | Waiting for table metadata lock| rename table t1 to t2|
  8. | 135 | test | Query   |       0 | init                           | show processlist     |
  9. +-----+------+---------+---------+--------------------------------+----------------------+
  10. 4 rows in set (0.00 sec)
  11. mysql> kill 134;
  12. Query OK, 0 rows affected (0.01 sec)
  13. #为了排版,表格字段略有删减,具体信息请看图片
复制代码

session2:session2重新连接,并且继续执行DDL语句,仍处于锁等待状态
  1. mysql> rename table t1 to t2;
  2. ERROR 2013 (HY000): Lost connection to MySQL server during query
  3. No connection. Trying to reconnect...
  4. Connection id:    136
  5. Current database: test
复制代码

session3:查看会话信息
  1. mysql> show processlist;
  2. +-----+------+---------+---------+--------------------------------+----------------------+
  3. | Id  | db   | Command | Time    | State                          | Info                 |
  4. +-----+------+---------+---------+--------------------------------+----------------------+
  5. |   6 | NULL | Daemon  | 4399260 | Waiting on empty queue         | NULL                 |
  6. | 132 | test | Sleep   |     479 |                                | NULL                 |
  7. | 135 | test | Query   |       0 | init                           | show processlist     |
  8. | 136 | test | Query   |     193 | Waiting for table metadata lock| rename table t1 to t2|
  9. +-----+------+---------+---------+--------------------------------+----------------------+
  10. 4 rows in set (0.00 sec)
  11. #为了排版,表格字段略有删减,具体信息请看图片
复制代码

可以看到, kill session2 后,session2 重新连接并且继续执行SQL
session1:提交事务
  1. mysql> commit;   
  2. Query OK, 0 rows affected (0.01 sec)  
复制代码

session2:执行成功
  1. mysql> use test
  2. Database changed
  3. mysql>
  4. mysql>
  5. mysql> rename table t1 to t2;
  6. ERROR 2013 (HY000): Lost connection to MySQL server during query
  7. No connection. Trying to reconnect...
  8. Connection id:    136
  9. Current database: testQuery OK, 0 rows affected (8 min 38.00 sec)
复制代码

通过上述测试,可以看到明明执行了 kill 命令,但是依然没有达到我们想要的效果,似乎 kill 命令没有生效一样。
经过查询资料发现,由于通过MySQL客户端登录,--reconnect 重新连接选项默认是开启的,该选项在每次连接丢失时都会进行一次重新连接尝试;因此在kill session2 后,session2重新连接并再次执行之前的SQL语句,导致感觉 kill 命令没有生效。
  1.   --reconnect         Reconnect if the connection is lost. Disable with
  2.                       --disable-reconnect. This option is enabled by default.
  3.                       (Defaults to on; use --skip-reconnect to disable.)
复制代码
解决

可以通过以下2种方式避免上述问题的发生:
1.执行kill query 命令

KILL QUERY终止连接当前正在执行的语句,但保持连接本身不变
session3:执行 KILL QUERY 命令
  1. mysql> show processlist;
  2. +-----+------+---------+---------+--------------------------------+----------------------+
  3. | Id  | db   | Command | Time    | State                          | Info                 |
  4. +-----+------+---------+---------+--------------------------------+----------------------+
  5. |   6 | NULL | Daemon  | 4401560 | Waiting on empty queue         | NULL                 |
  6. | 132 | test | Sleep   |      11 |                                | NULL                 |
  7. | 135 | test | Query   |       0 | init                           | show processlist     |
  8. | 137 | test | Query   |       3 | Waiting for table metadata lock| rename table t1 to t2|
  9. +-----+------+---------+---------+--------------------------------+----------------------+
  10. 4 rows in set (0.00 sec)
  11. mysql>
  12. mysql> kill query 137;
  13. Query OK, 0 rows affected (0.00 sec)
  14. #为了排版,表格字段略有删减,具体信息请看图片
复制代码

session2:
  1. mysql> rename table t1 to t2;   
  2. ERROR 1317 (70100): Query execution was interrupted  
复制代码

可以看到session2执行的语句已经被终止了,达到了我们想要的效果。
2.登录mysql客户端时加--skip-reconnect选项

--skip-reconnect 表示当连接丢失时不会进行重新连接的尝试
session2:登录时加 --skip-reconnect 选项
  1. shell> mysql -uroot -p -h127.0.0.1  -P3306 --skip-reconnect  
复制代码
session3:执行 kill 命令
  1. mysql> show processlist;
  2. +-----+------+---------+---------+--------------------------------+----------------------+
  3. | Id  | db   | Command | Time    | State                          | Info                 |
  4. +-----+------+---------+---------+--------------------------------+----------------------+
  5. |   6 | NULL | Daemon  | 4402073 | Waiting on empty queue         | NULL                 |
  6. | 132 | test | Sleep   |     524 |                                | NULL                 |
  7. | 135 | test | Query   |       0 | init                           | show processlist     |
  8. | 139 | test | Query   |       4 | Waiting for table metadata lock| rename table t1 to t2|
  9. +-----+------+---------+---------+--------------------------------+----------------------+
  10. 4 rows in set (0.00 sec)
  11. mysql> kill 139;
  12. Query OK, 0 rows affected (0.00 sec)
复制代码

session2:
  1. mysql> rename table t1 to t2;
  2. ERROR 2013 (HY000): Lost connection to MySQL server during query
复制代码

可以看到session2的会话连接已经被终止,并且没有自动重新连接,达到了我们想要的效果。
总结


  • 通过MySQL客户端登录时,会话重新连接的选项 --reconnect 默认是开启的,如果要禁止重新连接可在登录时添加 --skip-reconnect
  • KILL CONNECTION 与 KILL 相同,它在终止连接正在执行的任何语句后,再终止会话连接。
  • KILL QUERY 终止连接当前正在执行的语句,但保持连接本身不变。
参考链接
https://dev.mysql.com/doc/refman/8.0/en/kill.html
https://dev.mysql.com/doc/refman/8.0/en/mysql-command-options.html

Enjoy GreatSQL
来源:https://www.cnblogs.com/greatsql/p/17246625.html
免责声明:由于采集信息均来自互联网,如果侵犯了您的权益,请联系我们【E-Mail:cb@itdo.tech】 我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x

举报 回复 使用道具