MySQL中lock tables和unlock tables浅析

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS PostgreSQL,集群系列 2核4GB
简介: MySQL中lock tables和unlock tables浅析   在MySQL中提供了锁定表(lock tables)和解锁表(unlock tables)的语法功能,ORACLE与SQL Server数据库当中没有这种语法。

MySQL中lock tables和unlock tables浅析

 

MySQL中提供了锁定表(lock tables)和解锁表(unlock tables)的语法功能,ORACLESQL Server数据库当中没有这种语法。相信刚接触MySQL的人,都想详细、深入的了解一下这个功能.下面就尽量全面的解析、总结一下MySQLlock tablesunlock tables的功能,如有不足或不正确的地方,欢迎指点一二。

 

 

锁定表的语法:

 

LOCK TABLES

tbl_name [AS alias] {READ [LOCAL] | [LOW_PRIORITY] WRITE}

[, tbl_name [AS alias] {READ [LOCAL] | [LOW_PRIORITY] WRITE}] ...

 

 

 LOCAL修饰符表示可以允许在其他会话中对在当前会话中获取了READ锁的的表执行插入。但是当保持锁时,若使用Server外的会话来操纵数据库则不能使用READ LOCAL。另外,对于InnoDB表,READ LOCALREAD相同。

 

The LOCAL modifier enables nonconflicting INSERT statements (concurrent inserts) by other sessions to execute while the lock is held. (See Section 8.11.3, “Concurrent Inserts”.) However, READ LOCAL cannot be used if you are going to manipulate the database using processes external to the server while you hold the lock. For InnoDB tables, READ LOCAL is the same as READ.

 

修饰符LOW_PRIORITY用于之前版本的MySQL,它会影响锁定行为,但是从MySQL 5.6.5以后,这个修饰符已经被弃用。如果使用它则会产生警告。

 

 

[LOW_PRIORITY] WRITE lock:

 

The session that holds the lock can read and write the table.

 

Only the session that holds the lock can access the table. No other session can access it until the lock is released.

 

Lock requests for the table by other sessions block while the WRITE lock is held.

 

The LOW_PRIORITY modifier has no effect. In previous versions of MySQL, it affected locking behavior, but this is no longer true. As of MySQL 5.6.5, it is deprecated and its use produces a warning. Use WRITE without LOW_PRIORITY instead.

 

 

解锁表的语法:

 

UNLOCK TABLES

 

 

LOCK TABLES为当前会话锁定表。 UNLOCK TABLES释放被当前会话持有的任何锁。官方文档13.3.5 LOCK TABLES and UNLOCK TABLES Syntax已经对LOCK TALESUNLOCK  TABLES做了不少介绍,下面我们通过一些测试例子来深入的理解一下锁表与解锁表的相关知识点。我们先准备一下测试环境用的表和数据。

 

mysql> create table test( id int, name varchar(12));
Query OK, 0 rows affected (0.07 sec)
 
mysql> insert into test
    -> select 10001, 'kerry'   union all
    -> select 10002, 'richard' union all
    -> select 10003, 'jimmy' ;
Query OK, 3 rows affected (0.05 sec)
Records: 3  Duplicates: 0  Warnings: 0
 
mysql> 

 

当前会话(会话ID61)持有test表的READ锁后,那么当前会话只可以读该表,而不能往表中写入数据,否则就会报Table 'test' was locked with a READ lock and can't be updated这样的错误。

 

mysql> select connection_id();
+-----------------+
| connection_id() |
+-----------------+
|              61 |
+-----------------+
1 row in set (0.00 sec)
 
mysql> show open tables where in_use >=1;
Empty set (0.00 sec)
 
mysql> lock tables test read;
Query OK, 0 rows affected (0.00 sec)
 
mysql> show open tables where in_use >=1;
+----------+-------+--------+-------------+
| Database | Table | In_use | Name_locked |
+----------+-------+--------+-------------+
| MyDB     | test  |      1 |           0 |
+----------+-------+--------+-------------+
1 row in set (0.01 sec)
 
mysql> select * from test;
+-------+---------+
| id    | name    |
+-------+---------+
| 10001 | kerry   |
| 10002 | richard |
| 10003 | jimmy   |
+-------+---------+
3 rows in set (0.00 sec)
 
mysql> insert into test
    -> values(10004, 'ken');
ERROR 1099 (HY000): Table 'test' was locked with a READ lock and can't be updated
mysql> 

 

 

clip_image001

 

 

另外,我们测试一下修饰符LOCAL的用途,如下所示:

 

mysql> create table test2( id int , name varchar(12)) engine=MyISAM;
Query OK, 0 rows affected (0.05 sec)
 
mysql> insert into test2
    -> select 1001, 'test';
Query OK, 1 row affected (0.00 sec)
Records: 1  Duplicates: 0  Warnings: 0
mysql> select connection_id();
+-----------------+
| connection_id() |
+-----------------+
|              66 |
+-----------------+
1 row in set (0.00 sec)
 
mysql> lock tables test2 read local;
Query OK, 0 rows affected (0.00 sec)
 
mysql> select * from test2;
+------+------+
| id   | name |
+------+------+
| 1001 | test |
+------+------+
1 row in set (0.00 sec)
 
mysql> insert into test2
    -> select 1002, 'kkk';
ERROR 1099 (HY000): Table 'test2' was locked with a READ lock and can't be updated
mysql> 

 

在其它会话当中,你可以看到表test2可以被插入。当然前提是表的存储引擎不能是InnoDB引擎,否则使用修饰符LOCAL和不用LOCAL是一样的,其它会话无法对表写入。

 

mysql> select connection_id();
+-----------------+
| connection_id() |
+-----------------+
|              65 |
+-----------------+
1 row in set (0.00 sec)
 
mysql> select * from test2;
+------+------+
| id   | name |
+------+------+
| 1001 | test |
+------+------+
1 row in set (0.00 sec)
 
mysql> insert into test2
    -> select 1002, 'kkk';
Query OK, 1 row affected (0.00 sec)
Records: 1  Duplicates: 0  Warnings: 0
 

那么其他会话是否也能读此表呢?  其它会话能否也能锁定该表(LOCK TABLES READ)? 其它会话是否也能锁定该表呢?(LOCK TABLES WRITE

 

mysql> select connection_id();
+-----------------+
| connection_id() |
+-----------------+
|              62 |
+-----------------+
1 row in set (0.01 sec)
 
mysql> select * from test;
+-------+---------+
| id    | name    |
+-------+---------+
| 10001 | kerry   |
| 10002 | richard |
| 10003 | jimmy   |
+-------+---------+
3 rows in set (0.00 sec)
 
mysql> lock tables test read;
Query OK, 0 rows affected (0.00 sec)
 
mysql> show open tables where in_use >=1;
+----------+-------+--------+-------------+
| Database | Table | In_use | Name_locked |
+----------+-------+--------+-------------+
| MyDB     | test  |      2 |           0 |
+----------+-------+--------+-------------+
1 row in set (0.00 sec)
 
mysql> unlock tables;
Query OK, 0 rows affected (0.00 sec)
 
mysql> show open tables where in_use >=1;
+----------+-------+--------+-------------+
| Database | Table | In_use | Name_locked |
+----------+-------+--------+-------------+
| MyDB     | test  |      1 |           0 |
+----------+-------+--------+-------------+
1 row in set (0.00 sec)
 
mysql> lock tables test write;

 

 

clip_image002

 

 

如上测试所示,如果一个会话在一个表上获得一个READ锁后,该会话和所有其他会话只能从表中读。不能往表中写,其它会话也可在该表获取一个READ锁,此时你会在show open tables里面看到in_use的值增加。其实LOCK TABLES READ是一个表锁,而且是共享锁。但是当一个会话获取一个表上的READ锁后,其它会话就不能获取该表的WRITE锁了,此时就会被阻塞,直到持有READ锁的会话释放READ锁。

 

 

clip_image003

 

另外需要注意的是,当前会话如果锁定了其中一个表,那么是无法查询其它表的。否则会报ERROR 1100 (HY000): Table 'worklog' was not locked with LOCK TABLES错误。

 

clip_image004

 

 

那么我们再来看看WRITE锁吧。测试前,先在上面两个会话中执行 unlock tables命令。然后获得表TEST上的一个WRITE锁,如下所示,当前会话可以读写表TEST

 

mysql> unlock tables;
Query OK, 0 rows affected (0.00 sec)
 
mysql> select connection_id();
+-----------------+
| connection_id() |
+-----------------+
|              61 |
+-----------------+
1 row in set (0.00 sec)
 
mysql> show open tables where in_use >=1;
Empty set (0.00 sec)
 
mysql> lock tables test write;
Query OK, 0 rows affected (0.00 sec)
 
mysql> select * from test;
+-------+---------+
| id    | name    |
+-------+---------+
| 10001 | kerry   |
| 10002 | richard |
| 10003 | jimmy   |
+-------+---------+
3 rows in set (0.00 sec)
 
mysql> update test set name='ken' where id=10003;
Query OK, 1 row affected (0.01 sec)
Rows matched: 1  Changed: 1  Warnings: 0
 
mysql> 

 

其它会话无法读写表TEST,都会被阻塞,当然也无法获取表TESTREAD锁或WRITE锁。也就是说当一个会话获得一个表上的一个WRITE锁后,那么只有持锁的会话READWRITE表,其他会话都会被阻止。

 

mysql> unlock tables;
Query OK, 0 rows affected (0.00 sec)
 
mysql> 
mysql> 
mysql> show open tables where in_use >=1;
+----------+-------+--------+-------------+
| Database | Table | In_use | Name_locked |
+----------+-------+--------+-------------+
| MyDB     | test  |      1 |           0 |
+----------+-------+--------+-------------+
1 row in set (0.00 sec)
 
mysql> select * from test;

 

 

clip_image005

 

 

mysql> select connection_id();
+-----------------+
| connection_id() |
+-----------------+
|              63 |
+-----------------+
1 row in set (0.00 sec)
 
mysql> show processlist;
+----+------+-----------+------+---------+------+---------------------------------+--------------------+
| Id | User | Host      | db   | Command | Time | State                           | Info               |
+----+------+-----------+------+---------+------+---------------------------------+--------------------+
| 61 | root | localhost | MyDB | Sleep   |   86 |                                 | NULL               |
| 62 | root | localhost | MyDB | Query   |   40 | Waiting for table metadata lock | select * from test |
| 63 | root | localhost | MyDB | Query   |    0 | init                            | show processlist   |
| 64 | root | localhost | MyDB | Sleep   | 2551 |                                 | NULL               |
+----+------+-----------+------+---------+------+---------------------------------+--------------------+
4 rows in set (0.00 sec)

 

 

UNLOCK TABLES释放被当前会话持有的任何锁,但是当会话发出另外一个LOCK TABLES时,或当服务器的连接被关闭时,当前会话锁定的所有表会隐式被解锁。下面我们也可以测试看看

 

mysql> lock tables test read;
Query OK, 0 rows affected (0.00 sec)
 
mysql> show open tables where in_use >=1;
+----------+-------+--------+-------------+
| Database | Table | In_use | Name_locked |
+----------+-------+--------+-------------+
| MyDB     | test  |      1 |           0 |
+----------+-------+--------+-------------+
1 row in set (0.00 sec)
 
mysql> lock tables worklog read;
Query OK, 0 rows affected (0.00 sec)
 
mysql> show open tables where in_use >=1;
+----------+---------+--------+-------------+
| Database | Table   | In_use | Name_locked |
+----------+---------+--------+-------------+
| MyDB     | worklog |      1 |           0 |
+----------+---------+--------+-------------+
1 row in set (0.00 sec)
 
mysql> 

 

clip_image006

 

那么我们如何在当前会话锁定多个表呢?如下所示:

 

 

mysql> show open tables where in_use >=1;
Empty set (0.00 sec)
 
mysql> lock tables test read, worklog read;
Query OK, 0 rows affected (0.00 sec)
 
mysql> show open tables where in_use >=1;
+----------+---------+--------+-------------+
| Database | Table   | In_use | Name_locked |
+----------+---------+--------+-------------+
| MyDB     | worklog |      1 |           0 |
| MyDB     | test    |      1 |           0 |
+----------+---------+--------+-------------+
2 rows in set (0.00 sec)
 
mysql> 

 

 

 

另外,还有一些细节问题,LOCK TABLES是否可以为视图、触发器、临时表加锁呢?

 

 

mysql> create table test2( id int, sex bit);
Query OK, 0 rows affected (0.06 sec)
 
mysql> insert into test2
    -> select 10001, 1 union all
    -> select 10002, 0 union all
    -> select 10003, 1;
Query OK, 3 rows affected (0.02 sec)
Records: 3  Duplicates: 0  Warnings: 0
mysql> create view v_test
    -> as
    -> select t1.id, t1.name, t2.sex
    -> from test t1 left join test2 t2 on t1.id =t2.id;
Query OK, 0 rows affected (0.01 sec)
mysql> lock tables v_test read;
Query OK, 0 rows affected (0.00 sec)
 
mysql> show open tables where in_use >=1;
+----------+-------+--------+-------------+
| Database | Table | In_use | Name_locked |
+----------+-------+--------+-------------+
| MyDB     | test2 |      1 |           0 |
| MyDB     | test  |      1 |           0 |
+----------+-------+--------+-------------+
2 rows in set (0.00 sec)
 
mysql> 

 

 

 

如上测试所示,对于VIEW加锁,LOCK TABLES语句会为VIEW中使用的所有基表加锁。对触发器使用LOCK TABLE,那么就会锁定触发器中所包含的全部表(any tables used in triggers are also locked implicitly

 

 

mysql> unlock tables;

Query OK, 0 rows affected (0.00 sec)

 

mysql> create temporary table tmp like test;

Query OK, 0 rows affected (0.04 sec)

 

mysql> show open tables where in_use >=1;

Empty set (0.00 sec)

 

mysql> select database();

+------------+

| database() |

+------------+

| MyDB       |

+------------+

1 row in set (0.00 sec)

 

mysql> select * from tmp;

Empty set (0.00 sec)

 

mysql> insert into tmp

    -> select 1001, 'kerry' ;

Query OK, 1 row affected (0.01 sec)

Records: 1  Duplicates: 0  Warnings: 0

 

mysql>

 

LOCK TABLES UNLOCK TABLES只能为自己获取锁和释放锁,不能为其他会话获取锁,也不能释放由其他会话保持的锁。一个对象获取锁,需具备该对象上的SELECT权限和LOCK TABLES权限。LOCK TABLES语句为当前会话显式的获取表锁。最后,关于LOCK TABLES与事务当中锁有那些异同,可以参考官方文档13.3.5.1 Interaction of Table Locking and Transactions

 

LOCK TABLES and UNLOCK TABLES interact with the use of transactions as follows:

 

 

·         LOCK TABLES is not transaction-safe and implicitly commits any active transaction before attempting to lock the tables.

 

 

·         UNLOCK TABLES implicitly commits any active transaction, but only if LOCK TABLES has been used to acquire table locks. For example, in the following set of statements,UNLOCK TABLES releases the global read lock but does not commit the transaction because no table locks are in effect:

 

 

相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助     相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
3月前
|
关系型数据库 MySQL Java
MySQL数据锁:Record Lock,Gap Lock 和 Next-Key Lock
本文基于 MySQL 8.0.30 版本及 InnoDB 引擎,深入解析三种行锁机制:记录锁(Record Lock)、间隙锁(Gap Lock)和临键锁(Next-key Lock)。记录锁锁定索引记录,确保事务唯一修改;间隙锁锁定索引间的间隙,防止新记录插入;临键锁结合两者,锁定范围并记录自身,有效避免幻读现象。通过具体示例展示了不同锁的作用机制及其在并发控制中的应用。
360 2
|
7月前
|
存储 SQL 关系型数据库
【MySQL技术内幕】6.1-锁、lock和latch
【MySQL技术内幕】6.1-锁、lock和latch
106 0
|
关系型数据库 MySQL 数据库
MySQL报错:Lock wait timeout exceeded; try restarting transaction
MySQL报错:Lock wait timeout exceeded; try restarting transaction
744 0
|
SQL 关系型数据库 MySQL
MySQL报错:1205 Lock wait timeout exceeded; try restarting transaction处理
MySQL报错:1205 Lock wait timeout exceeded; try restarting transaction处理
427 0
|
SQL 关系型数据库 MySQL
【MySQL异常】MySQL事务锁问题----lock wait timeout exceeded; try restarting transaction
【MySQL异常】MySQL事务锁问题----lock wait timeout exceeded; try restarting transaction
533 0
|
SQL 关系型数据库 MySQL
Mysql 异常:Lock wait timeout exceeded; try restarting transaction的解决办法
Mysql 异常:Lock wait timeout exceeded; try restarting transaction的解决办法
498 0
|
SQL 关系型数据库 MySQL
mysql Lock wait timeout exceeded; try restarting transaction解决方案
在测试程序时,打的断点怎么都跳不进去,console一直报 “Lock wait timeout exceeded; try restarting transaction”
191 0
|
20天前
|
存储 Oracle 关系型数据库
数据库传奇:MySQL创世之父的两千金My、Maria
《数据库传奇:MySQL创世之父的两千金My、Maria》介绍了MySQL的发展历程及其分支MariaDB。MySQL由Michael Widenius等人于1994年创建,现归Oracle所有,广泛应用于阿里巴巴、腾讯等企业。2009年,Widenius因担心Oracle收购影响MySQL的开源性,创建了MariaDB,提供额外功能和改进。维基百科、Google等已逐步替换为MariaDB,以确保更好的性能和社区支持。掌握MariaDB作为备用方案,对未来发展至关重要。
45 3
|
20天前
|
安全 关系型数据库 MySQL
MySQL崩溃保险箱:探秘Redo/Undo日志确保数据库安全无忧!
《MySQL崩溃保险箱:探秘Redo/Undo日志确保数据库安全无忧!》介绍了MySQL中的三种关键日志:二进制日志(Binary Log)、重做日志(Redo Log)和撤销日志(Undo Log)。这些日志确保了数据库的ACID特性,即原子性、一致性、隔离性和持久性。Redo Log记录数据页的物理修改,保证事务持久性;Undo Log记录事务的逆操作,支持回滚和多版本并发控制(MVCC)。文章还详细对比了InnoDB和MyISAM存储引擎在事务支持、锁定机制、并发性等方面的差异,强调了InnoDB在高并发和事务处理中的优势。通过这些机制,MySQL能够在事务执行、崩溃和恢复过程中保持
47 3
|
20天前
|
SQL 关系型数据库 MySQL
数据库灾难应对:MySQL误删除数据的救赎之道,技巧get起来!之binlog
《数据库灾难应对:MySQL误删除数据的救赎之道,技巧get起来!之binlog》介绍了如何利用MySQL的二进制日志(Binlog)恢复误删除的数据。主要内容包括: 1. **启用二进制日志**:在`my.cnf`中配置`log-bin`并重启MySQL服务。 2. **查看二进制日志文件**:使用`SHOW VARIABLES LIKE 'log_%';`和`SHOW MASTER STATUS;`命令获取当前日志文件及位置。 3. **创建数据备份**:确保在恢复前已有备份,以防意外。 4. **导出二进制日志为SQL语句**:使用`mysqlbinlog`
63 2