一、前言

上一篇说了下innodb中锁的大概意思, 这篇说说怎么查看加的哪些锁。不然后续出现死锁或者锁等待都不知道为什么。

二、底层基础表信息

在学会如何查看有哪些锁信息时, 需要了解一些基础表信息, 这些能帮助我们快速排查。

从前两篇文章可以了解到innodb中的锁是在事务内执行的,所以我们先了解下底层的事务表看看从中可以看出哪些内容。

2.1 information_schema.INNODB_TRX

底层有两个databases

mysql> show databases;
+--------------------+
| Database |
+--------------------+
| information_schema |
| mysql |
| performance_schema |
| sys |
| test_db |
+--------------------+
5 rows in set (0.01 sec)

可以选择information_schema 查看下面是否有事务相关的表。

mysql> use information_schema;
Reading table information for completion of table and column names
You can turn off this feature to get a quicker startup with -A Database changed
mysql> show tables like '%tr%';
+-------------------------------------+
| Tables_in_information_schema (%TR%) |
+-------------------------------------+
| CHECK_CONSTRAINTS |
| INNODB_METRICS |
| INNODB_TRX |
| OPTIMIZER_TRACE |
| REFERENTIAL_CONSTRAINTS |
| ST_GEOMETRY_COLUMNS |
| TABLE_CONSTRAINTS |
| TRIGGERS |
+-------------------------------------+
8 rows in set (0.00 sec)

可见存在事务表INNODB_TRX, 然后看看其表结构,然后针对每个字段的解释加进去

mysql> show create table INNODB_TRX;
.... INNODB_TRX | CREATE TEMPORARY TABLE `INNODB_TRX` (
# 事务ID
`trx_id` varchar(18) NOT NULL DEFAULT '', # 事务状态, 允许值是 RUNNING,LOCK WAIT, ROLLING BACK,和 COMMITTING。
`trx_state` varchar(13) NOT NULL DEFAULT '', # 事务开始时间
`trx_started` datetime NOT NULL DEFAULT '0000-00-00 00:00:00', # 事务当前等待的锁的ID,如果TRX_STATE是LOCK WAIT;否则NULL。
`trx_requested_lock_id` varchar(105) DEFAULT NULL, # 事务开始等待锁的时间
`trx_wait_started` datetime DEFAULT NULL, # 事务权重, 反映(但不一定是准确计数)更改的行数和事务锁定的行数。为了解决死锁, InnoDB选择权重最小的事务作为“受害者”进行回滚。无论更改和锁定行的数量如何,更改非事务表的事务都被认为比其他事务更重。
`trx_weight` bigint(21) unsigned NOT NULL DEFAULT '0', # MySQL 线程 ID。 这个id很重要,如果发现某个事务一直在等待无法结束的话,可以通过这个ID kill掉。
`trx_mysql_thread_id` bigint(21) unsigned NOT NULL DEFAULT '0', # 事务正在执行的 SQL 语句。
`trx_query` varchar(1024) DEFAULT NULL, # 交易的当前操作,如果有的话;否则 NULL。
`trx_operation_state` varchar(64) DEFAULT NULL, # InnoDB处理此事务的当前 SQL 语句时使用 的表数。
`trx_tables_in_use` bigint(21) unsigned NOT NULL DEFAULT '0', # InnoDB当前 SQL 语句具有行锁 的表数。(因为这些是行锁,而不是表锁,尽管某些行被锁定,但通常仍可以由多个事务读取和写入表。)
`trx_tables_locked` bigint(21) unsigned NOT NULL DEFAULT '0', # 事务保留的锁数。
`trx_lock_structs` bigint(21) unsigned NOT NULL DEFAULT '0', # 此事务的锁结构在内存中占用的总大小。
`trx_lock_memory_bytes` bigint(21) unsigned NOT NULL DEFAULT '0', # 此事务锁定的大致数量或行数。该值可能包括物理上存在但对事务不可见的删除标记行。
`trx_rows_locked` bigint(21) unsigned NOT NULL DEFAULT '0', # 此事务中修改和插入的行数。
`trx_rows_modified` bigint(21) unsigned NOT NULL DEFAULT '0',
`trx_concurrency_tickets` bigint(21) unsigned NOT NULL DEFAULT '0', # 当前事务的隔离级别。
`trx_isolation_level` varchar(16) NOT NULL DEFAULT '',
`trx_unique_checks` int(1) NOT NULL DEFAULT '0',
`trx_foreign_key_checks` int(1) NOT NULL DEFAULT '0',
`trx_last_foreign_key_error` varchar(256) DEFAULT NULL,
`trx_adaptive_hash_latched` int(1) NOT NULL DEFAULT '0',
`trx_adaptive_hash_timeout` bigint(21) unsigned NOT NULL DEFAULT '0',
`trx_is_read_only` int(1) NOT NULL DEFAULT '0',
`trx_autocommit_non_locking` int(1) NOT NULL DEFAULT '0'
) ENGINE=MEMORY DEFAULT CHARSET=utf8 | 1 row in set (0.00 sec)

上述已经针对重要字段进行了注释说明,该表主要是记录事务中的一些信息,非常有用,其中就会记录等钱等待锁的ID。

详细请看官方文档:https://dev.mysql.com/doc/refman/8.0/en/information-schema-innodb-trx-table.html

其中以下几个字段需要特别留意下,

TRX_ID                  事务ID,
TRX_REQUESTED_LOCK_ID 事务当前等待的锁的ID。 如果当前事务阻塞就可以看出之前的锁
TRX_MYSQL_THREAD_ID MySQL 线程 ID

2.2 performance_schema.data_locks

上述事务表中有记录当前等待锁的ID, 那么这个id来源哪呢?

可以在information_schema performance_schema中搜索show tables like '%lock%';, 后面发现在performance_schema

mysql> show tables like '%lock%';
+---------------------------------------+
| Tables_in_performance_schema (%lock%) |
+---------------------------------------+
| data_lock_waits |
| data_locks |
| metadata_locks |
| rwlock_instances |
| table_lock_waits_summary_by_table |
+---------------------------------------+
5 rows in set (0.00 sec)

先看看data_locks的表结构:

CREATE TABLE `data_locks` (
# 持有或请求锁的存储引擎。
`ENGINE` varchar(32) NOT NULL, # 存储引擎持有或请求的锁的 ID。( ENGINE_LOCK_ID, ENGINE) 值的元组是唯一的。
# information_schema.INNODB_TRX.trx_requested_lock_id 就来源于这
`ENGINE_LOCK_ID` varchar(128) NOT NULL, # 请求锁定的事务的存储引擎内部 ID
# 来源information_schema.INNODB_TRX.TRX_ID
`ENGINE_TRANSACTION_ID` bigint(20) unsigned DEFAULT NULL, # 创建锁的会话的线程 ID
`THREAD_ID` bigint(20) unsigned DEFAULT NULL, `EVENT_ID` bigint(20) unsigned DEFAULT NULL,
`OBJECT_SCHEMA` varchar(64) DEFAULT NULL,
`OBJECT_NAME` varchar(64) DEFAULT NULL,
`PARTITION_NAME` varchar(64) DEFAULT NULL,
`SUBPARTITION_NAME` varchar(64) DEFAULT NULL, # 锁定索引的名称
`INDEX_NAME` varchar(64) DEFAULT NULL,
`OBJECT_INSTANCE_BEGIN` bigint(20) unsigned NOT NULL, # 锁的类型。该值取决于存储引擎。对于 InnoDB,允许的值为 RECORD行级锁和 TABLE表级锁。
`LOCK_TYPE` varchar(32) NOT NULL, # 如何请求锁定。
# 该值取决于存储引擎。为 InnoDB,允许值是 S[,GAP],X[,GAP], IS[,GAP],IX[,GAP], AUTO_INC,和 UNKNOWN。AUTO_INC和UNKNOWN 指示间隙锁定以外的锁定模式 (如果存在)
`LOCK_MODE` varchar(32) NOT NULL, # 锁定请求的状态。
# 该值取决于存储引擎。对于 InnoDB,允许的值为 GRANTED(锁定已持有)和 WAITING(正在等待锁定)。
`LOCK_STATUS` varchar(32) NOT NULL,
`LOCK_DATA` varchar(8192) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci DEFAULT NULL,
PRIMARY KEY (`ENGINE_LOCK_ID`,`ENGINE`),
KEY `ENGINE_TRANSACTION_ID` (`ENGINE_TRANSACTION_ID`,`ENGINE`),
KEY `THREAD_ID` (`THREAD_ID`,`EVENT_ID`),
KEY `OBJECT_SCHEMA` (`OBJECT_SCHEMA`,`OBJECT_NAME`,`PARTITION_NAME`,`SUBPARTITION_NAME`)
) ENGINE=PERFORMANCE_SCHEMA DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci

详细参数解释请参考: https://dev.mysql.com/doc/refman/8.0/en/performance-schema-data-locks-table.html

从上面可以知道当前事务如果持有锁的就看出它持有的什么类型的锁、锁状态。

三、实践得真知

1、开始一个事务1, 对某条记录加排他锁:

mysql> begin;
Query OK, 0 rows affected (0.00 sec) mysql> select * from t where id = 3 for update;
+----+------+
| id | name |
+----+------+
| 3 | 3 |
+----+------+
1 row in set (0.00 sec)

然后根据当前线程id查询事务信息:

mysql> select * from information_schema.INNODB_TRX where TRX_MYSQL_THREAD_ID = CONNECTION_ID() \G
*************************** 1. row ***************************
trx_id: 38441
trx_state: RUNNING
trx_started: 2021-08-22 09:26:56
trx_requested_lock_id: NULL
trx_wait_started: NULL
trx_weight: 2
trx_mysql_thread_id: 32
trx_query: select * from information_schema.INNODB_TRX where TRX_MYSQL_THREAD_ID = CONNECTION_ID()
trx_operation_state: NULL
trx_tables_in_use: 0
trx_tables_locked: 1
trx_lock_structs: 2
trx_lock_memory_bytes: 1136
trx_rows_locked: 1
trx_rows_modified: 0
trx_concurrency_tickets: 0
trx_isolation_level: REPEATABLE READ
trx_unique_checks: 1
trx_foreign_key_checks: 1
trx_last_foreign_key_error: NULL
trx_adaptive_hash_latched: 0
trx_adaptive_hash_timeout: 0
trx_is_read_only: 0
trx_autocommit_non_locking: 0
1 row in set (0.00 sec)

可以看出当前事务ID38441, 锁定行数为1行, 符合预期。

然后再根据事务ID查看锁信息:

mysql> select * from performance_schema.data_locks where ENGINE_TRANSACTION_ID = 38441 \G
*************************** 1. row ***************************
ENGINE: INNODB
ENGINE_LOCK_ID: 4720840032:1068:140354321295272
ENGINE_TRANSACTION_ID: 38441
THREAD_ID: 72
EVENT_ID: 246
OBJECT_SCHEMA: test_db
OBJECT_NAME: t
PARTITION_NAME: NULL
SUBPARTITION_NAME: NULL
INDEX_NAME: NULL
OBJECT_INSTANCE_BEGIN: 140354321295272
LOCK_TYPE: TABLE
LOCK_MODE: IX
LOCK_STATUS: GRANTED
LOCK_DATA: NULL
*************************** 2. row ***************************
ENGINE: INNODB
ENGINE_LOCK_ID: 4720840032:11:4:2:140354330466328
ENGINE_TRANSACTION_ID: 38441
THREAD_ID: 72
EVENT_ID: 246
OBJECT_SCHEMA: test_db
OBJECT_NAME: t
PARTITION_NAME: NULL
SUBPARTITION_NAME: NULL
INDEX_NAME: PRIMARY
OBJECT_INSTANCE_BEGIN: 140354330466328
LOCK_TYPE: RECORD
LOCK_MODE: X,REC_NOT_GAP
LOCK_STATUS: GRANTED
LOCK_DATA: 3
2 rows in set (0.00 sec)

可以看出当前事务对应两个锁信息, 第一个是表锁:意向排他锁, 第二个是行锁:排他锁 且 非间隙锁, 都是持有锁的状态, 而且锁的记录也是primarKey = 3的那条记录。 符合预期。

这里可能会有好奇,为啥会有表锁呢? 不熟悉的可以再看看之前的文章:https://www.cnblogs.com/yuanfy008/p/14993366.html

2、开始一个事务2, 先查看当前线程id, 然后对id=3的那条记录加排他锁。

mysql> begin;
mysql> select connection_id();
+-----------------+
| connection_id() |
+-----------------+
| 33 |
+-----------------+
1 row in set (0.00 sec) mysql> select * from t where id = 3 for update;

这是会锁等待, 因为事务1占有着呢。

然后再去另外一个窗口根据mysql线程id查看事务情况:

mysql> select * from information_schema.INNODB_TRX where TRX_MYSQL_THREAD_ID = 33 \G
*************************** 1. row ***************************
trx_id: 38445
trx_state: LOCK WAIT
trx_started: 2021-08-22 09:52:40
trx_requested_lock_id: 4720840880:11:4:2:140354330471280
trx_wait_started: 2021-08-22 09:55:56
trx_weight: 2
trx_mysql_thread_id: 33
trx_query: select * from t where id = 3 for update
trx_operation_state: starting index read
trx_tables_in_use: 1
trx_tables_locked: 1
trx_lock_structs: 2
trx_lock_memory_bytes: 1136
trx_rows_locked: 2
trx_rows_modified: 0
trx_concurrency_tickets: 0
trx_isolation_level: REPEATABLE READ
trx_unique_checks: 1
trx_foreign_key_checks: 1
trx_last_foreign_key_error: NULL
trx_adaptive_hash_latched: 0
trx_adaptive_hash_timeout: 0
trx_is_read_only: 0
trx_autocommit_non_locking: 0
1 row in set (0.00 sec)

可以看出当前事务还在等待另一个锁(ID:4720840880:11:4:2:140354330471280)的释放,而这个锁的持有这正好是事务1。符合预期

这个时候我们再去看这个事务对应锁的信息, 那这个时候有几把锁呢? 应该只有一把:表锁 - 意向排他锁

mysql> select * from performance_schema.data_locks where ENGINE_TRANSACTION_ID = 38445 \G
*************************** 1. row ***************************
ENGINE: INNODB
ENGINE_LOCK_ID: 4720840880:1068:140354321297272
ENGINE_TRANSACTION_ID: 38445
THREAD_ID: 73
EVENT_ID: 31
OBJECT_SCHEMA: test_db
OBJECT_NAME: t
PARTITION_NAME: NULL
SUBPARTITION_NAME: NULL
INDEX_NAME: NULL
OBJECT_INSTANCE_BEGIN: 140354321297272
LOCK_TYPE: TABLE
LOCK_MODE: IX
LOCK_STATUS: GRANTED
LOCK_DATA: NULL
1 row in set (0.00 sec)

Innodb中怎么查看锁信息的更多相关文章

  1. MySQL/InnoDB中,乐观锁、悲观锁、共享锁、排它锁、行锁、表锁、死锁概念的理解

    文章出处:https://www.souyunku.com/2018/07/30/mysql/?utm_source=tuicool&utm_medium=referral MySQL/Inn ...

  2. sqlserver检测死锁;杀死锁和进程;查看锁信息

    http://blog.sina.com.cn/s/blog_9dcdd2020101nf4v.html sqlserver检测死锁;杀死锁和进程;查看锁信息 ( ::)转载▼ 标签: sql 检测死 ...

  3. 查看锁信息(开启InnoDB监控)

    当前mysql版本:5.6.21 一.背景 在mysql处理死锁问题时,由于show engine innodb status输出来的死锁日志无任务事务上下文,并不能很好地诊断相关事务所持有的所有锁信 ...

  4. linux系统中如何查看acpi信息?

    答: 进入/sys/firmware/acpi/tables, 然后输入tree命令即可查看acpi信息

  5. 查看锁信息 v$lock 和 v$locked_object

    查看锁住的对象及会话id,serial# select a.*  from (SELECT o.object_name,               l.locked_mode,            ...

  6. SQL Server 2008中查看锁信息

    ;with tran_locks as(select resource_type,db_name(resource_database_id) as db_name,resource_descripti ...

  7. Innodb中的行锁与表锁

    在Innodb引擎中既支持行锁也支持表锁,那么什么时候会锁住整张表,什么时候或只锁住一行呢? InnoDB行锁是通过给索引上的索引项加锁来实现的,这一点MySQL与Oracle不同,后者是通过在数据块 ...

  8. MySQL/InnoDB中,对于锁的认识

    MySQL/InnoDB的加锁,一直是一个面试中常问的话题.例如,数据库如果有高并发请求,如何保证数据完整性?产生死锁问题如何排查并解决?我在工作过程中,也会经常用到,乐观锁,排它锁,等.于是今天就对 ...

  9. InnoDB的锁机制浅析(二)—探索InnoDB中的锁(Record锁/Gap锁/Next-key锁/插入意向锁)

    Record锁/Gap锁/Next-key锁/插入意向锁 文章总共分为五个部分: InnoDB的锁机制浅析(一)-基本概念/兼容矩阵 InnoDB的锁机制浅析(二)-探索InnoDB中的锁(Recor ...

随机推荐

  1. mysql 的基础操作

    1.建表 create table  表名( 字段一 数据类型 [列属性] , 字段二 数据类型 [列属性], ......... )[表类型][表字符集][注释]; 注意:MySQL命令终止符为分号 ...

  2. java基础---类和对象(1)

    一. 类和对象 面向对象:以属性和行为的观点去分析现实生活中的事物,将功能封装进对象, 强调具备了功能的对象,以类/对象为最小单位,考虑谁来做 面向过程:强调的是功能行为,以函数为最小单位,考虑怎么做 ...

  3. Java+Selenium3.3.1环境搭建

    一.背景和目的 selenium从2.0开始,加入了webdriver,实际上,我们说的selenium自动化测试,大部分情况都是在使用webdriver的API.现在去Selenium官网,发现最新 ...

  4. (精)题解 guP4878 [USACO05DEC] 布局

    差分约束模版题 不过后三个点简直是满满的恶意qwq 这里不说做题思路(毕竟纯模板),只说几个坑点: 1. 相邻的两头牛间必须建边(这点好像luogu没有体现),例如一组数据: 4 1 1 1 4 10 ...

  5. 「AGC010F」 Tree Game

    「AGC010F」 Tree Game 传送门 切了一个 AGC 的题,很有精神. 于是决定纪念一下. 首先如果任意一个人在点 \(u\),他肯定不会向点权大于等于 \(a_u\) 的点走的,因为此时 ...

  6. sql2008编辑前200行怎么修改

    打开Microsoft SQL Server Management Studio--工具菜单--选项---SQL Server对象资源管理器---命令--右侧"编辑前n行命令的值:1000

  7. 学习 CLR 源码:连续内存块数据操作的性能优化

    目录 C# 原语类型 1,利用 Buffer 优化数组性能 2,BinaryPrimitives 细粒度操作字节数组 提高代码安全性 3,BitConverter.MemoryMarshal 4,Ma ...

  8. Requests方法 --- json模块

    1.Json 简介:Json,全名 JavaScript Object Notation,是一种轻量级的数据交换格式,常用于 http 请求中 2.可以用 help(json),查看对应的源码注释内容 ...

  9. CentOS下配置Nginx实现动静分离实例

    测试环境: CentOS Linux release 7.6 PHP 7.2.32 两台服务器:192.168.1.109(Nginx),192.168.1.118(Apache) 1. 安装配置19 ...

  10. java命名规则/规范

    Java命名规则: 名称只能由字母.数字.下划线.$符号组成. 不能以数字开头,不能包含空格. 名称不能使用Java中的关键字. Java命名规范: 项目名全部小写:     project 包名全部 ...