MySQL锁等待排查的问题解决

 更新时间:2025年08月11日 11:30:17   作者:Bing@DBAVictor356  
本文主要介绍了MySQL如何排查锁等待问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

分析MySQL的锁等待问题有助于发现和解决数据库性能瓶颈。锁等待问题通常会导致数据库响应时间变长,影响系统的整体性能。以下是详细深入的方法和代码示例,帮助你分析和解决MySQL的锁等待问题。

一、锁的类型和概念

在MySQL中,主要有以下几种锁:

  1. 表锁(Table Lock):针对整张表的锁。
  2. 行锁(Row Lock):针对表中某一行的锁。
  3. 共享锁(S Lock,也称读锁):允许其他事务同时读,但不能写。
  4. 排他锁(X Lock,也称写锁):不允许其他事务读或写。

InnoDB存储引擎主要使用行锁,MyISAM存储引擎使用表锁。在分析锁等待问题时,重点关注InnoDB的行锁。

二、启用和配置慢查询日志

慢查询日志可以记录锁等待时间较长的查询。

2.1 编辑MySQL配置文件

my.cnfmy.ini文件中添加或修改以下配置:

[mysqld]
slow_query_log = 1
slow_query_log_file = /var/log/mysql/mysql-slow.log
long_query_time = 1  # 设置慢查询的阈值,单位是秒
log_queries_not_using_indexes = 1  # 记录未使用索引的查询(可选)

2.2 重启MySQL服务

sudo systemctl restart mysql  # 对于systemd系统
# 或者
sudo service mysql restart  # 对于init.d系统

三、动态监控锁等待

MySQL提供了几个动态监控锁等待的工具和视图。

3.1 使用SHOW ENGINE INNODB STATUS

该命令提供InnoDB存储引擎内部状态的详细信息,包括锁等待情况。

SHOW ENGINE INNODB STATUS;

输出的部分示例:

------------------------
LATEST DETECTED DEADLOCK
------------------------
2021-10-01 10:25:34 0x7f8b0c3e3700
*** (1) TRANSACTION:
TRANSACTION 123456, ACTIVE 5 sec inserting
mysql tables in use 1, locked 1
LOCK WAIT 4 lock struct(s), heap size 1136, 2 row lock(s)
MySQL thread id 10, OS thread handle 140236724795136, query id 123 localhost root update
INSERT INTO orders (order_id, customer_id, order_date) VALUES (1, 123, '2021-10-01')
*** (1) WAITING FOR THIS LOCK TO BE GRANTED:
RECORD LOCKS space id 163 page no 3 n bits 72 index PRIMARY of table `test`.`orders` trx id 123456 lock_mode X locks rec but not gap waiting
Record lock, heap no 2 PHYSICAL RECORD: n_fields 4; compact format; info bits 0

3.2 使用INFORMATION_SCHEMA表

INFORMATION_SCHEMA中有几个表可以提供锁等待相关的信息:

  1. INNODB_TRX:当前运行的事务。
  2. INNODB_LOCKS:当前持有的和等待的锁。
  3. INNODB_LOCK_WAITS:当前锁等待情况。
-- 查找当前正在等待的锁
SELECT
    r.trx_id waiting_trx_id,
    r.trx_mysql_thread_id waiting_thread,
    r.trx_query waiting_query,
    b.trx_id blocking_trx_id,
    b.trx_mysql_thread_id blocking_thread,
    b.trx_query blocking_query
FROM
    information_schema.innodb_lock_waits w
    JOIN information_schema.innodb_trx b ON b.trx_id = w.blocking_trx_id
    JOIN information_schema.innodb_trx r ON r.trx_id = w.requesting_trx_id;

示例输出:

+------------------+----------------+-----------------------------------+-----------------+----------------+-------------------------------------+
| waiting_trx_id   | waiting_thread | waiting_query                    | blocking_trx_id | blocking_thread| blocking_query                       |
+------------------+----------------+-----------------------------------+-----------------+----------------+-------------------------------------+
| CDC123456        | 8              | INSERT INTO orders...            | BAC987654       | 7              | UPDATE orders SET...                |
+------------------+----------------+-----------------------------------+-----------------+----------------+-------------------------------------+

四、分析和解决锁等待问题

根据收集到的信息,可以采取以下措施来解决锁等待问题:

4.1 优化查询和索引

确保查询使用了适当的索引,以减少锁的范围和时间。

-- 创建索引
CREATE INDEX idx_customer_id ON orders(customer_id);

4.2 分析和优化事务

减少事务的粒度,确保事务尽可能短,避免长时间持有锁。

BEGIN;
-- 执行一些操作
COMMIT;

4.3 调整隔离级别

根据业务需求调整事务隔离级别,以减少锁冲突。

-- 设置全局隔离级别
SET GLOBAL transaction_isolation='READ-COMMITTED';

-- 设置当前会话的隔离级别
SET SESSION transaction_isolation='READ-COMMITTED';

4.4 使用锁超时

设置锁等待超时时间,避免长时间等待锁。

[mysqld]
innodb_lock_wait_timeout = 50  # 设置锁等待超时时间,单位是秒

五、监控和调整

  1. 持续监控:使用监控工具(如Prometheus、Grafana、Percona Monitoring and Management)持续监控锁等待情况。
  2. 定期检查:定期检查锁等待日志和相关指标,及时发现和解决问题。
  3. 自动化调优:使用自动化调优工具(如MySQL Tuner、Percona Toolkit)定期进行优化。
# 使用MySQL Tuner
wget http://mysqltuner.pl/ -O mysqltuner.pl
chmod +x mysqltuner.pl
./mysqltuner.pl

六、总结

分析和解决MySQL锁等待问题需要综合利用慢查询日志、MySQL内部状态命令和INFORMATION_SCHEMA视图。通过优化查询和索引、分析和优化事务、调整隔离级别以及设置锁超时,可以有效减少锁等待问题。同时,持续监控和定期检查有助于及时发现并解决潜在的锁等待问题,从而提升数据库性能和稳定性。

到此这篇关于MySQL8锁等待排查的问题解决的文章就介绍到这了,更多相关MySQL 锁等待排查内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • MySQL 不等于的三种使用及区别

    MySQL 不等于的三种使用及区别

    MySQL中常用到判断符号,而不等于是比较常用的符号,不等于主要是三种,本文主要介绍了三种的使用及区别,感兴趣的同学可以了解一下
    2021-06-06
  • MySQL pt-slave-restart工具的使用简介

    MySQL pt-slave-restart工具的使用简介

    这篇文章主要介绍了MySQL pt-slave-restart工具的使用简介,帮助大家更好的理解和学习使用MySQL,感兴趣的朋友可以了解下
    2021-04-04
  • MySQL MyISAM默认存储引擎实现原理

    MySQL MyISAM默认存储引擎实现原理

    这篇文章主要介绍了MySQL MyISAM默认存储引擎实现原理,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-03-03
  • WITH在MYSQL中的用法示例详解

    WITH在MYSQL中的用法示例详解

    WITH 子句(也称为公共表表达式,Common Table Expression,简称 CTE)是 SQL 中一种强大的查询构建工具,它可以显著提高复杂查询的可读性和可维护性,这篇文章主要介绍了WITH在MYSQL中的用法,需要的朋友可以参考下
    2025-05-05
  • MySql中使用INSERT INTO语句更新多条数据的例子

    MySql中使用INSERT INTO语句更新多条数据的例子

    这篇文章主要介绍了MySql中使用INSERT INTO语句更新多条数据的例子,MySQL的特有语法,需要的朋友可以参考下
    2014-06-06
  • MySQL服务启动与停止的实现方式

    MySQL服务启动与停止的实现方式

    这篇文章主要介绍了MySQL服务启动与停止的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2025-06-06
  • MySQL字符串常用函数详解

    MySQL字符串常用函数详解

    本文给大家介绍MySQL字符串常用函数,本文结合实例代码给大家介绍的非常详细,对大家学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧
    2025-07-07
  • 深入分析MySQL Sending data查询慢问题

    深入分析MySQL Sending data查询慢问题

    给大家深入分析一下MySQL Sending data表查询慢的问题,并给出了详细的解决方案,一起来参考下。
    2017-12-12
  • 详解Mysql 30条军规

    详解Mysql 30条军规

    这篇文章主要介绍了详解Mysql 30条军规,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-08-08
  • 解析MySQL8.0新特性——事务性数据字典与原子DDL

    解析MySQL8.0新特性——事务性数据字典与原子DDL

    这篇文章主要介绍了MySQL8.0新特性——事务性数据字典与原子DDL的相关资料,帮助大家更好的理解和学习MySQL8.0感兴趣的朋友可以了解下
    2020-08-08

最新评论