死锁分析
1、死锁的产生
有以下代码,模拟的是两个账户之间的转账情况
void transfer(Account from,Account to,int money){
from.setAmount(from.getAmount()-money);
to.setAmount(to.getAmount()+money);
}
单线程下,这段代码肯定没问题的,但在多线程下存在问题
现在对其加锁,加锁后的代码如下
void transfer(Account from,Account to,int money){
synchronized(from){
synchronized(to){
from.setAmount(from.getAmount()-money);
to.setAmount(to.getAmount()+money);
}
}
}
synchronized,即对象锁,第一个synchronized锁住from对象,第二个synchronized锁住to对象。
多线程情况下 ,在同一时刻多个线程中仅有一个线程能拿到这个对象锁,对对象锁里面的代码段进行操作
但是加了对象锁之后,可能存在死锁!
transfer(a,b,100)和transfer(b,a,100)同时进行,即a向b转账的同时,b也在向a转账
【1】a向b转账,线程x拿到a这个对象锁
【2】b向a转账,线程y拿到b这个对象锁
操作【1】迫切需要b这个对象锁,才能进行转账操作
操作【2】迫切需要a这个对象锁,才能进行转账操作
两个线程都在等待该组中的其他线程释放锁,此时发生死锁!
2、死锁的解决
从那些方面下手呢?
【1】破除互斥等待
一般来说,我们为了程序的安全,必须给对象加上锁,所以这个条件一般无法破除
【2】破除hold and wait ,即占有等待
可以一次性获取所有的资源,示例代码中并不是一次性获取from和to这两个资源的,而是分布获取的。
方法一:给to加上一个很短的超时时间,一旦尝试获取to超时,则立即释放一开始持有的from锁,过一段时间后
再重新尝试获取from与to锁。是推荐的方法
方法二:给这段代码加上一个全局锁,可以保证from与to同时拿到,转账操作完成后,释放这个全局锁。是比较安全的,但是银行的账户数目众多,转账操作十分频繁,使用这种方式势必会造成性能的严重下降
【3】破除循环等待
按顺序获取资源,按照Account的id的大小进行转账的操作,id比较小的账户先进行转账操作,id比较大的账户后进行
转账操作。但现实生活中,有些事物是没有id的,即没有顺序性,这个时候需要强制给其加上顺序性。
【4】破除无法剥夺的等待
加入超时,不得已的方法,用户的此次转账操作失败,用户的体验不好
个人推荐的解法:
牺牲用户简短的等待时间,使用【2】中的方法1