时间:2021-07-01 10:21:17 帮助过:24人阅读
MySQL InnoDB存储引擎,实现的是基于多版本的并发控制协议——MVCC (Multi-Version Concurrency Control) (注:与MVCC相对的,是基于锁的并发控制,Lock-Based Concurrency Control)。MVCC最大的好处,相信也是耳熟能详:读不加锁,读写不冲突。在读多写少的OLTP应用中,读写不冲突是非常重要的,极大的增加了系统的并发性能,这也是为什么现阶段,几乎所有的RDBMS,都支持了MVCC。
在MVCC并发控制中,读操作可以分成两类:快照读 (snapshot read)与当前读 (current read)。快照读,读取的是记录的可见版本 (有可能是历史版本),不用加锁。当前读,读取的是记录的最新版本,并且,当前读返回的记录,都会加上锁,保证其他事务不会再并发修改这条记录。
简单的select操作,属于快照读,不加锁。(当然,也有例外,下面会分析)
select * from table where ?;
当前读:特殊的读操作,插入/更新/删除操作,属于当前读,需要加锁。
select * from table where ? lock in share mode;
S锁
select * from table where ? for update; X锁
insert into table values (…);X锁
update table set ? where ?;X锁
delete from table where ?;X锁
官方文档:
Every InnoDB table has a special index called the clustered index where the data for the rows is stored. Typically, the clustered index is synonymous with the primary key. To get the best performance from queries, inserts, and other database operations, you must understand how InnoDB uses the clustered index to optimize the most common lookup andDML
operations for each table.
If you define a
PRIMARY KEY
on your table, InnoDB uses it as the clustered index.(如果定义的主键,那么InnoDB会选择主键作为聚簇索引。)If you do not define a
PRIMARY KEY
for your table, MySQL picks the firstUNIQUE
index that has onlyNOT NULL
columns as the primary key andInnoDB
uses it as the clustered index.(如果不存在主键,那么Mysql会选择一个唯一建并且唯一建列为Not Null作为InnoDB索引。)If the table has no
PRIMARY KEY
or suitableUNIQUE
index,InnoDB
internally generates a hidden clustered index on a synthetic column containing rowID
values. The rows are ordered by theID
thatInnoDB
assigns to the rows in such a table. The rowID
is a 6-byte field that increases monotonically as new rows are inserted. Thus, the rows ordered by the rowID
are physically in insertion order.(如果表主键和唯一建都不存在,那么InnoDB内部会生成一个隐藏的包含了列号值,并且作为聚簇索引。)
在数据库和会话程序中,使用2PL来保证线程安全,即获取锁与释放锁。2PL有两个原语:
-Expanding phase : locks are acquired and no locks are released;
-Shrinking phase : locks are released and no locks are acquired;
2PL定于区分了两种锁,Shared locks 和 Exclusive locks。参见:Two-phase locking
在database systems中,isolations 决定事务对于其他用户和系统的可见性吗,PS:ACID(Atomicty,Consistency, Isolation, Durability)。
在一个事务的两次查询中数据笔数不一致,例如有一个事务查询了几行(Row)数据,而另一个事务却在此时插入了新的几行数据,先前的事务在接下来的查询中,就会发现有几行数据是它先前所没有的。
某个事务已更新一份数据,另一个事务在此时读取了同一份数据,由于某些原因,前一个RollBack了操作,则后一个事务所读取的数据就会是不正确的,原因是事务能看到别的事务的未提交的结果。
在一个事务的两次查询之中数据不一致,这可能是两次查询过程中间插入了一个事务更新的原有的数据,原因是事务可以看到别的事务提交的结果。
这是最高的隔离级别,它通过强制事务排序,使之不可能相互冲突,从而解决幻读问题。简言之,它是在每个读的数据行上加上共享锁。在这个级别,可能导致大量的超时现象和锁竞争。
这是MySQL的默认事务隔离级别,它确保同一事务的多个实例在并发读取数据时,会看到同样的数据行。不过理论上,这会导致另一个棘手的问题:幻读 (Phantom Read)。简单的说,幻读指当用户读取某一范围的数据行时,另一个事务又在该范围内插入了新行,当用户再读取该范围的数据行时,会发现有新的“幻影” 行。InnoDB和Falcon存储引擎通过多版本并发控制(MVCC,Multiversion Concurrency Control)机制解决了该问题。
这是大多数数据库系统的默认隔离级别(但不是MySQL默认的)。它满足了隔离的简单定义:一个事务只能看见已经提交事务所做的改变。这种隔离级别 也支持所谓的不可重复读(Nonrepeatable Read),因为同一事务的其他实例在该实例处理其间可能会有新的commit,所以同一select可能返回不同结果。
在该隔离级别,所有事务都可以看到其他未提交事务的执行结果。本隔离级别很少用于实际应用,因为它的性能也不比其他级别好多少。读取未提交的数据,也被称之为脏读(Dirty Read)。
何登成文章:http://hedengcheng.com/?p=771
mysql文档:http://dev.mysql.com/doc/
Mysql文章笔记
标签:事务隔离级别 锁 mysql 数据库