万答#1,MySQL中如何查询某个表上的IS(意向共享)锁

欢迎来到 GreatSQL社区分享的MySQL技术文章,如有疑问或想学习的内容,可以在下方评论区留言,看到后会进行解答

问题

问题原文是这样的:

假如在MySQL事务里,给某个表的一行加了 共享锁,理论上这个表本身会自动加上意向共享锁,那么能不能用 sql 查出这个表加了意向锁?

回答

答案是肯定的,当然可以执行SQL查询表上的IS锁加锁状态。

先声明,我们本次讨论的是MySQL里的InnoDB引擎表,下面讨论的内容都是基于这个前提。

在揭晓答案之前,多介绍点InnoDB引擎锁相关的一些知识吧。主要有以下几点

  • InnoDB引擎表既支持表级锁,也支持行级锁。

  • 加表级锁的方法和MyISAM表是一样的,执行 LOCK TABLE READ/WRITE 即可。

  • InnoDB表的行锁是加在索引上的,因此如果没有合适的索引,是会导致表里所有记录都被加上行锁,其后果等同于表级锁,但产生的影响比表级锁可就大多了。因为锁对象数量大了很多,消耗的内存也多很多。

  • 加上行锁时,同时还需要对表加上相应的意向锁。例如,想要对一行数据加上共享锁(S锁),则相应的要对表加上意向共享锁(IS锁);同样地,想要对一行数据加上排他锁(X锁),则相应的要对表加上意向排他锁(IX锁)。

  • 意向锁是加在聚集索引的根节点上的,因此无论锁定多少行,只需要加一个意向锁。

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

file

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

file

查询结果例如下面这样:

[root@yejr.run] [(none)]>select * from performance_schema.data_locks\G
*************************** 1. row ***************************
               ENGINE: INNODB
       ENGINE_LOCK_ID: 140701134495048:1350:140701396637648
ENGINE_TRANSACTION_ID: 422176111205704
            THREAD_ID: 87
             EVENT_ID: 95
        OBJECT_SCHEMA: yejr
          OBJECT_NAME: t1
       PARTITION_NAME: NULL
    SUBPARTITION_NAME: NULL
           INDEX_NAME: NULL
OBJECT_INSTANCE_BEGIN: 140701396637648
            LOCK_TYPE: TABLE
            LOCK_MODE: IS
          LOCK_STATUS: GRANTED
            LOCK_DATA: NULL
*************************** 2. row ***************************
               ENGINE: INNODB
       ENGINE_LOCK_ID: 140701134495048:267:4:9:140701409130528
ENGINE_TRANSACTION_ID: 422176111205704
            THREAD_ID: 87
             EVENT_ID: 95
        OBJECT_SCHEMA: yejr
          OBJECT_NAME: t1
       PARTITION_NAME: NULL
    SUBPARTITION_NAME: NULL
           INDEX_NAME: PRIMARY
OBJECT_INSTANCE_BEGIN: 140701409130528
            LOCK_TYPE: RECORD
            LOCK_MODE: S,REC_NOT_GAP
          LOCK_STATUS: GRANTED
            LOCK_DATA: 1

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

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

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

- session2查询PFS.data_locks
[root@yejr.run] [(none)]>select * from performance_schema.data_locks\G
*************************** 1. row ***************************
               ENGINE: INNODB
       ENGINE_LOCK_ID: 140701134495888:1350:140701396639728
ENGINE_TRANSACTION_ID: 104536
            THREAD_ID: 89
             EVENT_ID: 43
        OBJECT_SCHEMA: yejr
          OBJECT_NAME: t1
       PARTITION_NAME: NULL
    SUBPARTITION_NAME: NULL
           INDEX_NAME: NULL
OBJECT_INSTANCE_BEGIN: 140701396639728
            LOCK_TYPE: TABLE
            LOCK_MODE: IX  <-- 这个就是IX锁了
          LOCK_STATUS: GRANTED
            LOCK_DATA: NULL
*************************** 2. row ***************************
               ENGINE: INNODB
       ENGINE_LOCK_ID: 140701134495888:267:4:9:140701409135136
ENGINE_TRANSACTION_ID: 104536
            THREAD_ID: 89
             EVENT_ID: 43
        OBJECT_SCHEMA: yejr
          OBJECT_NAME: t1
       PARTITION_NAME: NULL
    SUBPARTITION_NAME: NULL
           INDEX_NAME: PRIMARY
OBJECT_INSTANCE_BEGIN: 140701409135136
            LOCK_TYPE: RECORD
            LOCK_MODE: X,REC_NOT_GAP
          LOCK_STATUS: GRANTED
            LOCK_DATA: 1

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

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

- session2查询表上有哪些MDL锁
[root@yejr.run] [(none)]>select * from performance_schema.metadata_locks\G
*************************** 1. row ***************************
          OBJECT_TYPE: TABLE
        OBJECT_SCHEMA: yejr
          OBJECT_NAME: t1
          COLUMN_NAME: NULL
OBJECT_INSTANCE_BEGIN: 140701215694512
            LOCK_TYPE: SHARED_READ  <- 共享读锁,可以同时加多个共享行锁
        LOCK_DURATION: TRANSACTION
          LOCK_STATUS: GRANTED
               SOURCE: sql_parse.cc:5761
      OWNER_THREAD_ID: 87
       OWNER_EVENT_ID: 100

也看下加排他行锁:

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

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

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

测试环境

Server version: 8.0.23 MySQL Community Server - GPL

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

Enjoy MySQL :)

文章推荐:

技术分享 | MGR最佳实践(MGR Best Practice) https://mp.weixin.qq.com/s/66u5K7a9u8GcE2KPn4kCaA

技术分享 | 万里数据库MGR Bug修复之路 https://mp.weixin.qq.com/s/IavpeP93haOKVBt7eO8luQ

Macos系统编译percona及部分函数在Macos系统上运算差异 https://mp.weixin.qq.com/s/jAbwicbRc1nQ0f2cIa_2nQ

技术分享 | 利用systemd管理MySQL单机多实例 https://mp.weixin.qq.com/s/iJjXwd0z1a6isUJtuAAHtQ

产品 | GreatSQL,打造更好的MGR生态 https://mp.weixin.qq.com/s/ByAjPOwHIwEPFtwC5jA28Q

产品 | GreatSQL MGR优化参考 https://mp.weixin.qq.com/s/5mL_ERRIjpdOuONian8_Ow

关于 GreatSQL

GreatSQL是由万里数据库维护的MySQL分支,专注于提升MGR可靠性及性能,支持InnoDB并行查询特性,是适用于金融级应用的MySQL分支版本。

Gitee: https://gitee.com/GreatSQL/GreatSQL

GitHub: https://github.com/GreatSQL/GreatSQL

微信&QQ群:

可扫码添加GreatSQL社区助手微信好友,发送验证信息“加群”加入GreatSQL/MGR交流微信群,亦可直接扫码加入GreatSQL/MGR交流QQ群。

file

本文由博客一文多发平台 OpenWrite 发布!

<think>我们正在讨论MySQL意向。根据引用内容,意向,用于指示事务稍后将对中的行施加哪种类型的共享或排他)。这样,当另一个事务想要定整个时,它可以快速检查意向而无需扫描每一行。 用户的问题:如何启用意向? 在MySQL中,意向是自动管理的,不需要显式启用。InnoDB存储引擎在需要时自动使用意向。具体来说: - 当您对中的行加共享(S)时,InnoDB会首先在上加一个意向共享IS)。 - 当您对中的行加排他(X)时,InnoDB会首先在上加一个意向排他(IX)。 因此,作为用户,您不需要(也无法)显式启用意向意向是InnoDB内部用于提高检查效率的机制。 但是,您需要确保您使用的是InnoDB存储引擎,因为意向是InnoDB的特性。MyISAM存储引擎不支持行级,也不支持意向。 另外,当您使用诸如`LOCK TABLES ... WRITE`这样的语句时,MySQL会获取。在InnoDB中,通常我们使用行级,但有时也会使用。在获取时,MySQL会检查是否有意向存在,从而避免与行级冲突。 总结:意向是InnoDB自动管理的,不需要用户显式启用。用户只需要确保使用InnoDB引擎。 根据引用[1]和[2],意向的作用是避免在加时检查每一行是否有,而是通过级的意向来快速判断。 因此,回用户:MySQL中的意向是InnoDB存储引擎自动管理的,不需要用户显式启用。只要您的使用InnoDB引擎,意向机制就会自动工作。 但是,用户可能想知道如何确保意向被使用?案是:只要使用InnoDB引擎,并且事务中使用了行级(如SELECT ... FOR UPDATE或SELECT ... LOCK IN SHARE MODE),InnoDB就会自动在上设置意向。 另外,关于意向查看,用户可以通过以下命令查看当前的状态: `SHOW ENGINE INNODB STATUS;` 或者在MySQL 5.7及以上版本,可以通过`performance_schema`中的查看,如`performance_schema.data_locks`。 下面,我们按照要求生成相关问题。 §§相关问题§§ 1. 意向在什么情况下会被使用? 2. 如何查看MySQL中的意向? 3. 意向和行之间的关系是什么? 4. 使用MyISAM存储引擎时,意向是否可用?</think>### MySQL 意向的启用机制 在 MySQL 中,**意向是 InnoDB 存储引擎自动管理的机制,不需要手动启用**。只要使用 InnoDB 引擎(MySQL 5.5 后的默认引擎),意向就会在以下场景自动生效: 1. **自动触发条件**: - 当事务对某行加**共享**(S )时,InnoDB 会先在级加**意向共享IS)**。 - 当事务对某行加**排他**(X )时,InnoDB 会先在级加**意向排他(IX)**。 - 例如执行 `SELECT ... LOCK IN SHARE MODE` 或 `SELECT ... FOR UPDATE` 时[^1][^2]。 2. **核心作用**: - **优化冲突检测**:当另一事务尝试加(如 `LOCK TABLES ... WRITE`)时,只需检查意向IS/IX),无需逐行扫描状态,大幅提升性能[^1][^2]。 - **避免死**:通过意向提前声明定意图,减少冲突概率。 3. **验证方法**: ```sql -- 步骤1:创建 InnoDB CREATE TABLE t ( id INT PRIMARY KEY, data VARCHAR(20) ) ENGINE=InnoDB; -- 步骤2:事务A 加行(自动触发意向) BEGIN; SELECT * FROM t WHERE id=1 FOR UPDATE; -- 级自动加 IX -- 步骤3:事务B 尝试加(被阻塞) LOCK TABLES t WRITE; -- 因检测到 IX 而等待 ``` 通过 `SHOW ENGINE INNODB STATUS` 可查看状态,输出中包含 `TABLE LOCK ... IX` 即意向生效[^1][^4]。 --- ### 关键注意事项 1. **存储引擎依赖**:仅 **InnoDB** 支持意向,MyISAM 等引擎无此机制。 2. **升级规则**:意向不会阻塞除全外的其他操作(如不同行的 IX 可共存)[^3]。 3. **与间隙协作**:意向与间隙(Gap Lock)协同工作,确保范围查询的隔离性[^3][^4]。 > 📌 **总结**:意向是 InnoDB 内置的自动化机制,只需确保使用 InnoDB 引擎并正常执行加操作(如 `FOR UPDATE`),系统会自动管理意向的申请与释放[^1][^2]。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值