暂无图片
暂无图片
暂无图片
暂无图片
暂无图片

MySQL:关于 RR 的一些实验及拓展(二)

原创 Ryan Bai 2024-05-08
119

从简单的增删改查角度,了解了一下锁,那么从视图角度是怎样的呢?

概念

锁的类型

  • 共享锁(S Lock):允许拥有共享锁的事务读取该行行数据。 当一个事务拥有一行的共享锁时,另外的事务可以在同一行数据也获得共享锁, 但另外
    的事务无法获得同一行数据上的排他锁
  • 排他锁(X Lock):允许拥有排它锁的事务修改或删除该行数据。 当一个事务拥有一行的排他锁时, 另外的事务在此行数据上无法获得共享锁和排它锁,只能等待第一个事务的锁释放
  • 除了共享锁和排他锁之外, InnoDB 也支持意图锁。 该锁类型是属于表级锁,表明事务在后期会对该表的行施加共享锁或者排它锁。
    • 意向共享锁(IS Lock):事务想要获得一张表中某几行的共享锁
    • 意向排他锁(IX Lock):事务想要获得一张表中某几行的排他锁

兼容性

IS IX S X
IS 兼容 兼容 兼容 不兼容
IX 兼容 兼容 不兼容 不兼容
S 兼容 不兼容 兼容 不兼容
X 不兼容 不兼容 不兼容 不兼容

疑问

  1. RR 锁有没有可能真的像其中一位大佬分析的那样,是行锁?(附上图,便于说明)
    image2024050615502379517149818799041.png
  2. 那 5 行数据查到的锁数据会不会也出现 5 行?(5 行是为了便于观察结果)

带着这些疑问,我做了如下实验

实验

准备

下面我做几个实验,从各个角度来分析一下 RR 和锁

  1. 先创建一张表,并插入测试数据

    create table t1 (id int auto_increment, a varchar(10), b varchar(10), c varchar(10), primary key(id) ) engine=INNODB default charset=utf8; DELIMITER $$ CREATE PROCEDURE InsertRows() BEGIN DECLARE i INT DEFAULT 1; WHILE i <= 5 DO INSERT INTO t1 (id, a, b, c) VALUES (i, CONCAT('Data', i), CONCAT('Data', i), CONCAT('Data', i)); SET i = i + 1; END WHILE; END$$ DELIMITER ; CALL InsertRows();

实验一

  1. 修改隔离级别

    set global transaction_isolation='repeatable-read';

    然后所有会话需要重新登录

  2. 确认隔离级别

    mysql> show variables like '%iso%'; +-----------------------+-----------------+ | Variable_name | Value | +-----------------------+-----------------+ | transaction_isolation | REPEATABLE-READ | +-----------------------+-----------------+ 1 row in set (0.00 sec)
  3. 会话一:开启事务,修改一条数据

    begin; update t1 set c='Data11' where a='Data5';
  4. 会话二:查看各个视图情况

    • table_handles

      mysql> select * from performance_schema.table_handles\G; *************************** 1. row *************************** OBJECT_TYPE: TABLE OBJECT_SCHEMA: test OBJECT_NAME: t1 OBJECT_INSTANCE_BEGIN: 139946276421760 OWNER_THREAD_ID: NULL OWNER_EVENT_ID: NULL INTERNAL_LOCK: NULL EXTERNAL_LOCK: NULL 1 row in set (0.00 sec)
    • metadata_locks

      mysql> select * from performance_schema.metadata_locks\G; *************************** 1. row *************************** OBJECT_TYPE: TABLE OBJECT_SCHEMA: test OBJECT_NAME: t1 COLUMN_NAME: NULL OBJECT_INSTANCE_BEGIN: 139946276255856 LOCK_TYPE: SHARED_WRITE LOCK_DURATION: TRANSACTION LOCK_STATUS: GRANTED SOURCE: sql_parse.cc:6093 OWNER_THREAD_ID: 48 OWNER_EVENT_ID: 5057 *************************** 2. row *************************** OBJECT_TYPE: TABLE OBJECT_SCHEMA: performance_schema OBJECT_NAME: metadata_locks COLUMN_NAME: NULL OBJECT_INSTANCE_BEGIN: 139946085863232 LOCK_TYPE: SHARED_READ LOCK_DURATION: TRANSACTION LOCK_STATUS: GRANTED SOURCE: sql_parse.cc:6093 OWNER_THREAD_ID: 50 OWNER_EVENT_ID: 38 2 rows in set (0.00 sec)
    • INNODB_TRX

      mysql> select * from information_schema.INNODB_TRX\G; *************************** 1. row *************************** trx_id: 3893 trx_state: RUNNING trx_started: 2024-05-07 11:22:15 trx_requested_lock_id: NULL trx_wait_started: NULL trx_weight: 3 trx_mysql_thread_id: 8 trx_query: NULL trx_operation_state: NULL trx_tables_in_use: 0 trx_tables_locked: 1 trx_lock_structs: 2 trx_lock_memory_bytes: 1128 trx_rows_locked: 6 trx_rows_modified: 1 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 trx_schedule_weight: NULL 1 row in set (0.00 sec)
    • data_locks

      mysql> select * from performance_schema.data_locks;

      image20240507112351698.png

结论:从上面截图可以知道,对于表的确会生成表 IX 锁,也会生成 5 行的行锁。还加了基数据锁,以防止有人对表结构做修改。

疑问:那么锁表的操作是 X 行锁还是 IX 表锁呢?

实验二

我们再来对比一下 RC 下和 RR 下有什么不同

  1. 修改隔离级别

    set global transaction_isolation='read-committed';

    然后所有会话需要重新登录

  2. 确认隔离级别

    mysql> show variables like '%iso%'; +-----------------------+----------------+ | Variable_name | Value | +-----------------------+----------------+ | transaction_isolation | READ-COMMITTED | +-----------------------+----------------+ 1 row in set (0.00 sec)
  3. 会话一:开启事务,修改一条数据

    begin; update t1 set c='Data11' where a='Data5';
  4. 会话二:查看各个视图情况

    • table_handles

      mysql> select * from performance_schema.table_handles\G; *************************** 1. row *************************** OBJECT_TYPE: TABLE OBJECT_SCHEMA: test OBJECT_NAME: t1 OBJECT_INSTANCE_BEGIN: 140324434622688 OWNER_THREAD_ID: NULL OWNER_EVENT_ID: NULL INTERNAL_LOCK: NULL EXTERNAL_LOCK: NULL *************************** 2. row *************************** OBJECT_TYPE: TABLE OBJECT_SCHEMA: test OBJECT_NAME: t1 OBJECT_INSTANCE_BEGIN: 140324434430400 OWNER_THREAD_ID: NULL OWNER_EVENT_ID: NULL INTERNAL_LOCK: NULL EXTERNAL_LOCK: NULL 2 rows in set (0.00 sec)
    • metadata_locks

      mysql> select * from performance_schema.metadata_locks\G; *************************** 1. row *************************** OBJECT_TYPE: TABLE OBJECT_SCHEMA: performance_schema OBJECT_NAME: metadata_locks COLUMN_NAME: NULL OBJECT_INSTANCE_BEGIN: 140324500471696 LOCK_TYPE: SHARED_READ LOCK_DURATION: TRANSACTION LOCK_STATUS: GRANTED SOURCE: sql_parse.cc:6093 OWNER_THREAD_ID: 50 OWNER_EVENT_ID: 4 *************************** 2. row *************************** OBJECT_TYPE: TABLE OBJECT_SCHEMA: test OBJECT_NAME: t1 COLUMN_NAME: NULL OBJECT_INSTANCE_BEGIN: 140324434446240 LOCK_TYPE: SHARED_WRITE LOCK_DURATION: TRANSACTION LOCK_STATUS: GRANTED SOURCE: sql_parse.cc:6093 OWNER_THREAD_ID: 49 OWNER_EVENT_ID: 15 2 rows in set (0.00 sec)
    • INNODB_TRX

      mysql> select * from information_schema.INNODB_TRX\G; *************************** 1. row *************************** trx_id: 4369 trx_state: RUNNING trx_started: 2024-05-07 11:40:55 trx_requested_lock_id: NULL trx_wait_started: NULL trx_weight: 3 trx_mysql_thread_id: 9 trx_query: NULL trx_operation_state: NULL trx_tables_in_use: 0 trx_tables_locked: 1 trx_lock_structs: 2 trx_lock_memory_bytes: 1128 trx_rows_locked: 1 trx_rows_modified: 1 trx_concurrency_tickets: 0 trx_isolation_level: READ COMMITTED 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 trx_schedule_weight: NULL 1 row in set (0.00 sec)
    • data_locks

      mysql> select * from performance_schema.data_locks;

      image20240507114234108.png

  5. 为了证明锁是表锁,我修改了一条数据,尴尬的是,修改成功。

    mysql> update t1 set c='Data11' where id=4; Query OK, 1 row affected (0.00 sec) Rows matched: 1 Changed: 1 Warnings: 0

结论:RC 下,对于表也会生成表 IX 锁和单行的无 GAP 行锁;还有基数据锁,依旧是为了防止有人对表结构做修改。

第一行非表排他锁,而是表意向排他,起到排他作用的锁是 X,整体看 RR 下是全表数据的 行排他+GAP锁 起到了表锁的现象。

总结

整体看,RC 环境下不会出现锁表的情况,而 RR 环境下的锁会更严一些。对于 UPDATE,会在 RC 的基础上对全表每一条数据增加 X 锁和 GAP 锁来达到锁表的目的,以保证不可虚读。

「喜欢这篇文章,您的关注和赞赏是给作者最好的鼓励」
关注作者
【版权声明】本文为墨天轮用户原创内容,转载时必须标注文章的来源(墨天轮),文章链接,文章作者等基本信息,否则作者和墨天轮有权追究责任。如果您发现墨天轮中有涉嫌抄袭或者侵权的内容,欢迎发送邮件至:contact@modb.pro进行举报,并提供相关证据,一经查实,墨天轮将立刻删除相关内容。

评论