在数据库中经常会遇到这样的情况:一个主表a,一个子表b,b表中包含有a表的主键作为外键。当要插入数据的时候,我们会先插入a表,然后获得a表的identity,再插入b表。如果要进行删除操作,那么就先删除子表b,然后再删除主表a。在程序设计中,对两个表的操作是在一个事务之中完成的。
当系统使用频繁就会出现插入操作和删除操作同时进行的情况。这个时候插入事务会先将主表a放置独占锁,然后去访问子表b,而同时删除事务会对子表b放置独占锁,然后去访问主表a。插入事务会一直独占着a表,等待访问b表,删除事务也一直独占着b表等待访问a表,于是两个事务相互独占一个表,等待对方释放资源,这样就造成了死锁。
遇到这种情况我听说了三种做法:
1 取消ab两个表之间的外键关系,这样就可以在删除数据的时候就可以先删除主表a,然后删除子表b,让对这两个表操作的事务访问顺序一致。
2 删除a表数据之前,先使用一个事务将b表中相关外键指向另外a表中的另外一个数据(比如在a表中专门建一行数据,主键设置为0,永远不会对这行数据执行删除操作),这样就消除了要被删除的数据在ab两个表中的关系。然后就可以使用删除事务,先删除a表中的数据,再删除b表中的数据,以达到和插入事务表访问一致,避免死锁。
3 在外键关系中,将“删除规则”设置为“层叠”,这样删除事务只需要直接去删除主表a,而不需要对子表b进行操作。因为删除规则设置为层叠以后,删除主表中的数据,子表中所有外键关联的数据也同时删除了。
以上三个解决办法都是同事给出的建议,我也不知道到底该使用什么办法才好。
不知道对于这种情况要防止死锁大家还有没有什么其他好办法?