死锁的处理
如果一个操作需要获取多个锁,例如锁A和锁B,那么对多个锁的获取应该始终有固定的先后顺序。例如总是先获取A锁,再获取B锁,否则可能会出现死锁。例如线程1先获取了A锁,线程2先获取了B锁,然后线程1等待获取B锁,线程2等待获取A锁,这样死锁就产生了。
有层次关系的资源的锁处理
如果资源有层级关系,一种简单的方法是只设置一个锁,无论什么操作都获取这个锁。这样正确性是可以保证,只是某些情况下可能并发性不好。于是可以对于不同层次的资源设置不同的锁。假设我们的层次关系如下:
A(锁)--> B(锁) --> C
A(锁)--> D
A资源下有B、D两种资源,其中A和B有自己的锁,其它资源没有自己的锁。我们以读写锁为例说明:
- 如果对C进行写操作,那么对A加读锁,对B加写锁。
- 如果对C进行读操作,那么对A加读锁,对B加读锁。
- 如果对D进行写操作,那么对A加写锁。
- 如果对D进行读操作,那么对A加读锁。
D没有自己的锁,所以依赖它上层资源的锁来保护。
C没有自己的锁,所以依赖它上层资源的锁来保护。而B的上层还有资源,所以在修改B下资源的时候,一般不会对B的上层加写锁,而是加读锁,以提高并发度。
异步锁
如果锁的区间非常大,并且需要长时间拿锁,那么可以考虑使用异步锁。即注册回调函数,说明想要获取的锁类型,然后等待回调函数通知。
锁的区间
除了异步锁之外,锁的区间应该尽量地短。因此,不允许在锁内进行内存分配、释放、打印日志等相关的操作。