GreatSQL社区

搜索

MySQL中如何查询某个表上的IS(意向共享)锁?

2743 1 2022-8-5 15:43
本帖最后由 GreatSQL万答 于 2022-8-16 15:31 编辑

假如在MySQL事务里,给某个表的一行加了共享锁,理论上这个表本身会自动加上意向共享锁,那么能不能用 sql 查出这个表加了意向锁?
全部回复(1)
GreatSQL万答 2022-8-5 15:44:52
本帖最后由 GreatSQL万答 于 2022-8-17 17:26 编辑

答案是肯定的,当然可以执行SQL查询表上的IS锁加锁状态。先声明,我们本次讨论的是MySQL里的InnoDB引擎表,下面讨论的内容都是基于这个前提。在揭晓答案之前,多介绍点InnoDB引擎锁相关的一些知识吧。主要有以下几点:

  • InnoDB引擎表既支持表级锁,也支持行级锁。
  • 加表级锁的方法和MyISAM表是一样的,执行 LOCK TABLE READ/WRITE 即可。
  • InnoDB表的行锁是加在索引上的,因此如果没有合适的索引,是会导致表里所有记录都被加上行锁,其后果等同于表级锁,但产生的影响比表级锁可就大多了。因为锁对象数量大了很多,消耗的内存也多很多。
  • 加上行锁时,同时还需要对表加上相应的意向锁。例如,想要对一行数据加上共享锁(S锁),则相应的要对表加上意向共享锁(IS锁);同样地,想要对一行数据加上排他锁(X锁),则相应的要对表加上意向排他锁(IX锁)。
  • 意向锁是加在聚集索引的根节点上的,因此无论锁定多少行,只需要加一个意向锁。

下面是几个锁之间的兼容矩阵


IS
IX
S
X
IS+++-
IX++--
S+-+-
X----

好了,接下来我们来看下怎么查看表级IS锁。其实很简单,只需要查看 PFS.data_locks 表就可以了。另一个表 PFS.metadata_locks 表可以查看MDL锁的详情。

session 1
session 2
begin;
-- 先加上共享行锁,此时也会对t1表加上IS锁
select * from t1 where c1=1 for share;


-- 观察到IS锁
select * from performance_schema.data_locks;

查询结果例如下面这样:

  1. [root@yejr.run] [(none)]>select * from performance_schema.data_locks\G
  2. *************************** 1. row ***************************
  3.                ENGINE: INNODB
  4.        ENGINE_LOCK_ID: 140701134495048:1350:140701396637648
  5. ENGINE_TRANSACTION_ID: 422176111205704
  6.             THREAD_ID: 87
  7.              EVENT_ID: 95
  8.         OBJECT_SCHEMA: yejr
  9.           OBJECT_NAME: t1
  10.        PARTITION_NAME: NULL
  11.     SUBPARTITION_NAME: NULL
  12.            INDEX_NAME: NULL
  13. OBJECT_INSTANCE_BEGIN: 140701396637648
  14.             LOCK_TYPE: TABLE
  15.             LOCK_MODE: IS
  16.           LOCK_STATUS: GRANTED
  17.             LOCK_DATA: NULL
  18. *************************** 2. row ***************************
  19.                ENGINE: INNODB
  20.        ENGINE_LOCK_ID: 140701134495048:267:4:9:140701409130528
  21. ENGINE_TRANSACTION_ID: 422176111205704
  22.             THREAD_ID: 87
  23.              EVENT_ID: 95
  24.         OBJECT_SCHEMA: yejr
  25.           OBJECT_NAME: t1
  26.        PARTITION_NAME: NULL
  27.     SUBPARTITION_NAME: NULL
  28.            INDEX_NAME: PRIMARY
  29. OBJECT_INSTANCE_BEGIN: 140701409130528
  30.             LOCK_TYPE: RECORD
  31.             LOCK_MODE: S,REC_NOT_GAP
  32.           LOCK_STATUS: GRANTED
  33.             LOCK_DATA: 1
复制代码

此时我们能看到t1表上共有两个锁,一个是表级IS锁,另一个是c1=1上的共享锁。

同样地,我们也可以观察IX锁或其他锁。

  1. - session1执行下面的SQL
  2. [root@yejr.run] [yejr]>begin; update t1 set c4=rand()*1024 where c1=1;

  3. - session2查询PFS.data_locks
  4. [root@yejr.run] [(none)]>select * from performance_schema.data_locks\G
  5. *************************** 1. row ***************************
  6.                ENGINE: INNODB
  7.        ENGINE_LOCK_ID: 140701134495888:1350:140701396639728
  8. ENGINE_TRANSACTION_ID: 104536
  9.             THREAD_ID: 89
  10.              EVENT_ID: 43
  11.         OBJECT_SCHEMA: yejr
  12.           OBJECT_NAME: t1
  13.        PARTITION_NAME: NULL
  14.     SUBPARTITION_NAME: NULL
  15.            INDEX_NAME: NULL
  16. OBJECT_INSTANCE_BEGIN: 140701396639728
  17.             LOCK_TYPE: TABLE
  18.             LOCK_MODE: IX  <-- 这个就是IX锁了
  19.           LOCK_STATUS: GRANTED
  20.             LOCK_DATA: NULL
  21. *************************** 2. row ***************************
  22.                ENGINE: INNODB
  23.        ENGINE_LOCK_ID: 140701134495888:267:4:9:140701409135136
  24. ENGINE_TRANSACTION_ID: 104536
  25.             THREAD_ID: 89
  26.              EVENT_ID: 43
  27.         OBJECT_SCHEMA: yejr
  28.           OBJECT_NAME: t1
  29.        PARTITION_NAME: NULL
  30.     SUBPARTITION_NAME: NULL
  31.            INDEX_NAME: PRIMARY
  32. OBJECT_INSTANCE_BEGIN: 140701409135136
  33.             LOCK_TYPE: RECORD
  34.             LOCK_MODE: X,REC_NOT_GAP
  35.           LOCK_STATUS: GRANTED
  36.             LOCK_DATA: 1
复制代码

进一步,我们简单看下MDL锁。加共享行锁:

  1. - session1加一个共享行锁
  2. [root@yejr.run] [yejr]>begin; select * from t1 where c1=1 for share;

  3. - session2查询表上有哪些MDL锁
  4. [root@yejr.run] [(none)]>select * from performance_schema.metadata_locks\G
  5. *************************** 1. row ***************************
  6.           OBJECT_TYPE: TABLE
  7.         OBJECT_SCHEMA: yejr
  8.           OBJECT_NAME: t1
  9.           COLUMN_NAME: NULL
  10. OBJECT_INSTANCE_BEGIN: 140701215694512
  11.             LOCK_TYPE: SHARED_READ  <- 共享读锁,可以同时加多个共享行锁
  12.         LOCK_DURATION: TRANSACTION
  13.           LOCK_STATUS: GRANTED
  14.                SOURCE: sql_parse.cc:5761
  15.       OWNER_THREAD_ID: 87
  16.        OWNER_EVENT_ID: 100
复制代码

也看下加排他行锁:

  1. - session1加一个排他行锁
  2. [root@yejr.run] [yejr]>begin; update t1 set c4=rand()*1024 where c1=1;

  3. - session2查询表上有哪些MDL锁
  4. [root@yejr.run] [(none)]>select * from performance_schema.metadata_locks\G
  5. *************************** 1. row ***************************
  6.           OBJECT_TYPE: TABLE
  7.         OBJECT_SCHEMA: yejr
  8.           OBJECT_NAME: t1
  9.           COLUMN_NAME: NULL
  10. OBJECT_INSTANCE_BEGIN: 140701215694640
  11.             LOCK_TYPE: SHARED_WRITE  <- 共享写锁,可以同时加多个排他行锁(不同数据行)
  12.         LOCK_DURATION: TRANSACTION
  13.           LOCK_STATUS: GRANTED
  14.                SOURCE: sql_parse.cc:5761
  15.       OWNER_THREAD_ID: 89
  16.        OWNER_EVENT_ID: 43
复制代码

好了,方法已有,更多的情形可以自己去玩了 :)


测试环境

Server version: 8.0.23 MySQL Community Server - GPL

上述PFS查看行锁、MDL锁的功能应该是8.0以上就开始支持了。

GreatSQL万答

22

主题

0

博客

73

贡献

版主

Rank: 7Rank: 7Rank: 7

金币
201
贡献
73

合作电话:010-64087828

社区邮箱:greatsql@greatdb.com

社区公众号
社区小助手
QQ群
GMT+8, 2025-1-18 16:05 , Processed in 0.020447 second(s), 14 queries , Redis On.
快速回复 返回顶部 返回列表