<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
锁是计算机协调多个进程或线程并发访问某一资源的机制。在数据库中,除传统的计算资源(如CPU、RAM、I/O等)的争用以外,数据也是一种供许多用户共享的资源。如何保证数据并发访问的一致性、有效性是所有数据库必须解决的一个问题,锁冲突也是影响数据库并发访问性能的一个重要因素。从这个角度来说,锁对数据库而言显得尤其重要,也更加复杂。本章我们着重讨论MySQL锁机制的特点,常见的锁问题,以及解决MySQL锁问题的一些方法或建议。
</div>
<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
MySQL锁概述
</div>
<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
相对其他数据库而言,MySQL的锁机制比较简单,其最显著的特点是不同的存储引擎支持不同的锁机制。比如,MyISAM和MEMORY存储引擎采用的是表级锁(table-level locking);BDB存储引擎采用的是页面锁(page-level locking),但也支持表级锁;InnoDB存储引擎既支持行级锁(row-level locking),也支持表级锁,但默认情况下是采用行级锁。
</div>
<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
MySQL这3种锁的特性可大致归纳如下。
</div>
<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
<span style=""><span style="margin:0px; padding:0px; font-weight:400; color:rgb(255,102,0)">开销、加锁速度、死锁、粒度、并发性能</span></span>
</div>
<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
l 表级锁:开销小,加锁快;不会出现死锁;锁定粒度大,发生锁冲突的概率最高,并发度最低。
</div>
<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
l 行级锁:开销大,加锁慢;会出现死锁;锁定粒度最小,发生锁冲突的概率最低,并发度也最高。
</div>
<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
l 页面锁:开销和加锁时间界于表锁和行锁之间;会出现死锁;锁定粒度界于表锁和行锁之间,并发度一般。
</div>
<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
从上述特点可见,很难笼统地说哪种锁更好,只能就具体应用的特点来说哪种锁更合适!仅从锁的角度来说:表级锁更适合于以查询为主,只有少量按索引条件更新数据的应用,如Web应用;而行级锁则更适合于有大量按索引条件并发更新少量不同数据,同时又有并发查询的应用,如一些在线事务处理(OLTP)系统。这一点在本书的“开发篇”介绍表类型的选择时,也曾提到过。下面几节我们重点介绍MySQL表锁和 InnoDB行锁的问题,由于BDB已经被InnoDB取代,即将成为历史,在此就不做进一步的讨论了。
</div>
<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
<br style="">
</div>
<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
MyISAM表锁
</div>
<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
MyISAM存储引擎只支持表锁,这也是MySQL开始几个版本中唯一支持的锁类型。随着应用对事务完整性和并发性要求的不断提高,MySQL才开始开发基于事务的存储引擎,后来慢慢出现了支持页锁的BDB存储引擎和支持行锁的InnoDB存储引擎(实际 InnoDB是单独的一个公司,现在已经被Oracle公司收购)。但是MyISAM的表锁依然是使用最为广泛的锁类型。本节将详细介绍MyISAM表锁的使用。
</div>
<h3 style="font-family:Arial; font-weight:100; line-height:26px; color:rgb(69,69,69); margin:0px; font-size:1.7em; padding:0px"> <a name="t0" style="background:transparent; color:rgb(79,161,219); margin:0px; padding:0px; outline:0px" target="_blank"></a><a name="t0" style="background:transparent; color:rgb(255,153,0); margin:0px; padding:0px; outline:0px" target="_blank"></a>查询表级锁争用情况</h3>
<div align="left" style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
可以通过检查table_locks_waited和table_locks_immediate状态变量来分析系统上的表锁定争夺:
</div>
<div style="margin:0px; padding:0px; color:rgb(69,69,69); font-family:Arial; font-size:14px; line-height:26px">
mysql>
<span style=""><span style="margin:0px; padding:0px; font-weight:400; color:rgb(25 |
|