锁是计算机协调多个进程或线程并发访问某一资源的机制。
相对其他数据库而言,MySQL的锁机制比较简单,其最 显著的特点是不同的存储引擎支持不同的锁机制。比如,MyISAM和MEMORY存储引擎采用的是表级锁(table-level locking);InnoDB存储引擎既支持行级锁(row-level locking),也支持表级锁,但默认情况下是采用行级锁。
MySQL的表级锁有两种模式:表共享读锁(Table Read Lock)和表独占写锁(Table Write Lock)。
表的读操作不会阻塞其他用户对同一表的读请求,但会阻塞对同一表的写请求;
对 MyISAM表的写操作,则会阻塞其他用户对同一表的读和写操作;
MyISAM表的读操作与写操作之间,以及写操作之间是串行的!
表读锁:lock table mylock read;
表写锁:lock table mylock write;
释放锁:unlock tables;
MyISAM在执行查询语句之前,会自动给涉及的所有表加读锁,在执行更新操作前,会自动给涉及的表加写锁,这个过程并不需要用户干预,因此用户一般不需要使用命令来显式加锁。
session1 | session2 |
---|---|
获取表的read local锁定lock table mylock read local |
|
当前session不能对表进行更新或者插入操作 insert into mylock values(6,‘f’) Table ‘mylock’ was locked with a READ lock and can’t be updated update mylock set name=‘aa’ where id = 1; Table ‘mylock’ was locked with a READ lock and can’t be updated |
其他session可以查询该表的记录 select* from mylock |
当前session不能查询没有锁定的表 select * from person Table ‘person’ was not locked with LOCK TABLES |
其他session可以进行插入操作,但是更新会阻塞 update mylock set name = ‘aa’ where id = 1; |
当前session不能访问其他session插入的记录; | |
释放锁资源:unlock tables |
当前session获取锁,更新操作完成 |
当前session可以查看其他session插入的记录 |
可以通过检查table_locks_waited和table_locks_immediate状态变量来分析系统上的表锁定争夺:
mysql> show status like ‘table%’;
±----------------------±------+
| Variable_name | Value |
±----------------------±------+
| Table_locks_immediate | 352 |
| Table_locks_waited | 2 |
±----------------------±------+
如果Table_locks_waited的值比较高,则说明存在着较严重的表级锁争用情况。
LOW_PRIORITY_UPDATES
参数,或在INSERT
、UPDATE
、DELETE
语句中指定LOW_PRIORITY
选项来调节读写锁的争用。select … lock in share mode
select … for update
InnoDB行锁是通过给索引上的索引项加锁来实现的,这一点MySQL与Oracle不同,后者是通过在数据块中对相应数据行加锁来实现的。
InnoDB这种行锁实现特点意味着:只有通过索引条件检索数据,InnoDB才使用行级锁,否则,InnoDB将使用表锁!
create table tab_no_index(id int,name varchar(10)) engine=innodb;
insert into tab_no_index values(1,'1'),(2,'2'),(3,'3'),(4,'4');
create table tab_with_index(id int,name varchar(10)) engine=innodb;
alter table tab_with_index add index id(id);
insert into tab_with_index values(1,'1'),(2,'2'),(3,'3'),(4,'4');
可以通过检查InnoDB_row_lock状态变量来分析系统上的行锁的争夺情况:
mysql> show status like ‘innodb_row_lock%’;
±------------------------------±------+
| Variable_name | Value |
±------------------------------±------+
| Innodb_row_lock_current_waits | 0 |
| Innodb_row_lock_time | 18702 |
| Innodb_row_lock_time_avg | 18702 |
| Innodb_row_lock_time_max | 18702 |
| Innodb_row_lock_waits | 1 |
±------------------------------±------+
如果发现锁争用比较严重,如InnoDB_row_lock_waits
和InnoDB_row_lock_time_avg
的值比较高