mysql中锁的介绍

来源:互联网 发布:mac助手 编辑:程序博客网 时间:2024/06/01 22:55

在数据库的锁机制中,按照锁的粒度划分,数据库锁可以分为行级锁(INNODB引擎),表级锁(MYISAM引擎),页级锁(BDB引擎)


锁:

   在计算机科学中,锁在执行多线程中用于强行限制资源访问的同步机制,用于在并发控制中保证对互斥要求的满足


各种常见锁的介绍:

行级锁:

  行级锁是mysql中锁定粒度最小的一种锁,只对当前操作的行进行加锁,行级锁能大大减少数据库操作的冲突,加锁粒度最小,但是加锁开销大!!!

  行级锁分为共享锁和排他锁

  特点:是加锁慢,开销大,会出现死锁,锁定粒度最小,发生锁冲突的概率也是最小,并发度最高

表级锁:

  表级锁是mysql中锁定粒度最大的一种锁,对当前操作的整张表都要加锁,实现比较简单,资源消耗比较少,大部分mysql引擎支持,常用的MYISAM和INNODB都支持表级锁定

  表级锁定分为表共享读锁(共享锁)和表独占写锁(排他锁)

  特点:开销小,加锁快,不会出现死锁,加锁粒度大,发生所冲突的概率高,并发度低


页级锁:

   业级锁是是mysql锁住粒度介于表级锁和行级锁之间的一种锁,表级锁速度快,但是冲突多,行级锁冲突少,但是速度慢,所以取了折衷的页级,一次锁定相邻的一组记录。BDB支持页级锁

   特点:开销和加锁时间介于表锁和行锁之间,会出现死锁;锁定粒度介于两者之间,并发度一般


MySQL常用存储引擎的锁机制

MyISAM和MEMORY采用表级锁(table-level locking)

BDB采用页面锁(page-level locking)或表级锁,默认为页面锁

InnoDB支持行级锁(row-level locking)和表级锁,默认为行级锁

Innodb中的行锁与表锁

前面提到过,在Innodb引擎中既支持行锁也支持表锁,那么什么时候会锁住整张表,什么时候或只锁住一行呢?

InnoDB行锁是通过给索引上的索引项加锁来实现的,这一点MySQL与Oracle不同,后者是通过在数据块中对相应数据行加锁来实现的。InnoDB这种行锁实现特点意味着:只有通过索引条件检索数据,InnoDB才使用行级锁,否则,InnoDB将使用表锁!

在实际应用中,要特别注意InnoDB行锁的这一特性,不然的话,可能导致大量的锁冲突,从而影响并发性能。

行级锁都是基于索引的,如果一条SQL语句用不到索引是不会使用行级锁的,会使用表级锁。行级锁的缺点是:由于需要请求大量的锁资源,所以速度慢,内存消耗大。

行级锁与死锁


MyISAM中是不会产生死锁的,因为MyISAM总是一次性获得所需的全部锁,要么全部满足,要么全部等待。而在InnoDB中,锁是逐步获得的,就造成了死锁的可能。

在MySQL中,行级锁并不是直接锁记录,而是锁索引。索引分为主键索引和非主键索引两种,如果一条sql语句操作了主键索引,MySQL就会锁定这条主键索引;如果一条语句操作了非主键索引,MySQL会先锁定该非主键索引,再锁定相关的主键索引。 在UPDATE、DELETE操作时,MySQL不仅锁定WHERE条件扫描过的所有索引记录,而且会锁定相邻的键值,即所谓的next-key locking。

当两个事务同时执行,一个锁住了主键索引在等待非主键索引,一个锁定了非主键索引,在等待主键索引。这样就会发生死锁。

发生死锁后,InnoDB一般都可以检测到,并使一个事务释放锁回退,另一个获取锁完成事务。

有多种方法可以避免死锁,这里只介绍常见的三种

1、如果不同程序会并发存取多个表,尽量约定以相同的顺序访问表,可以大大降低死锁机会。

2、在同一个事务中,尽可能做到一次锁定所需要的所有资源,减少死锁产生概率;

3、对于非常容易产生死锁的业务部分,可以尝试使用升级锁定颗粒度,通过表级锁定来减少死锁产生的概率;


0 0
原创粉丝点击