关于数据库死锁

不锁怕出事,锁了又怕锁死了!!!

数据库由于数据存储速度快,数据稳定,结构化的特性,被广泛用作数据存储,并成为最重要,最常见的方式!

数据库从20世纪50年代诞生伊始,就因为支持事务的特性得到大力的发展,最终各种数据库诸如oracle,Sybase,mysql等关系型数据库百花齐放,既然数据库是因为事务而生,那么事务的特性又是哪些呢?

简而言之就是ACID(原子性,一致性,隔离性,持久性)! 而为了保持数据的一致性,数据库都有了一个操作叫做加锁!有表级锁,行级锁,页面锁!

死锁就是说两个线程在争同一个资源,然后互不相让,导致锁死的情况(比如两个人从两端走上独木桥,然后卡死在桥中间)!

回到问题本身,<span>数据库在什么时候会产生死锁呢?

1,情况一:我中有你,你中有我!

事务一:两个操作update A;update B;

事务二:两个操作update B;update A;

线程一执行事务一到一半的时候,锁了A想要获得B的锁,与此同时事务二执行到了锁B,想要获得锁A的时候,因为互相都想要对方拥有的锁,而导致死锁!

2,情况二:吃着碗里的,看着锅里的!

A线程先查询了一条记录(使用了共享锁),与此同时B线程正要修改这条记录(使用了独占锁),然后A线程突然想修改这条记录了,怎么办呢?升级锁。。

而B线程想要降级为共享锁,必须要等到A线程释放掉共享锁,这样就形成了死锁!可以看到这个过程中是A占着共享锁想要升级,B占着独占锁想要降级,然后卡死!

3,牵一发而动全身!

一个表结构,必须要有适当的索引等优化手段,如果在执行事务的时候,没有加索引条件甚至没有任何条件,那么将执行全表扫描,如果是多个事务在操作,很容易就发生了阻塞和死锁!所以字段加索引非常重要!!

  数据库死锁的解决方案。

1)以固定的顺序访问表和行。即按顺序申请锁,这样就不会造成互相等待的场面。

2)大事务拆小。大事务更倾向于死锁,如果业务允许,将大事务拆小。

3)在同一个事务中,尽可能做到一次锁定所需要的所有资源,减少死锁概率。

4)降低隔离级别。如果业务允许,将隔离级别调低也是较好的选择,比如将隔离级别从RR调整为RC,可以避免掉很多因为gap锁造成的死锁。

5)为表添加合理的索引。如果不走索引将会为表的每一行记录添加上锁,死锁的概率大大增大。

上一篇:《DDIA》读书笔记:事务


下一篇:c#常用快捷键