线程死锁及解决办法

本文详细探讨了死锁的概念,指出死锁产生的四个必要条件:互斥、不可剥夺、请求与保持、循环等待。通过代码示例展示了死锁情况,并提出了预防、避免、检测和解除死锁的具体方法,包括一次性分配资源、避免循环等待、银行家算法、限时加锁等策略。
摘要由CSDN通过智能技术生成

什么是死锁

多线程以及多进程改善了系统资源的利用率并提高了系统的处理能力。然而为了保证同步,也会引入锁机制,并发执行也带来了新的问题–死锁。
所谓死锁是指多个线程因竞争资源而造成的一种僵局(互相等待),若无外力作用,这些进程都将无法向前推进。
在这里插入图片描述

死锁产生的必要条件

以下这四个条件是死锁的必要条件,只要系统发生死锁,这些条件必然成立,而只要上述条件之一不满足,就不会发生死锁。

  1. 互斥条件
    进程要求对所分配的资源(如打印机)进行排他性控制(当一个线程请求打印机时其他线程不能操作的),即在一段时间内某资源仅为一个进程所占有。此时若有其他进程请求该资源,则请求进程只能等待。
  2. 不可剥夺条件 进程所获得的资源在未使用完毕之前,不能被其他进程强行夺走,即只能由获得该资源的进程自己来释放(只能是主动释放)。
  3. 请求与保持条件
    进程已经保持了至少一个资源,但又提出了新的资源请求,而该资源已被其他进程占有,此时请求进程被阻塞,但对自己已获得的资源保持不放。
  4. 循环等待条件 存在一种进程资源的循环等待链,链中每一个进程已获得的资源同时被
    链中下一个进程所请求。即存在一个处于等待状态的进程集合{Pl, P2, …, pn},其中Pi等 待的资源被P(i+1)占有(i=0,
    1, …, n-1),Pn等待的资源被P0占有,如图所示。
    在这里插入图片描述在这里插入图片描述

死锁示例代码

以下情况无法打印第二次请求的资源,因为上一个线程没有释放资源,所以陷入死锁

public class DeadLockRunnable implements Runnable {
   

    private int flag;//决定线程走向的标记
    //需要注意这两个对象一定要是两个实例共享的
    //不然每次new的就会是两个独立对象
    //因此需要把每一个Object对象设置为静态对象static
    private static Object obj1 = new Object();//锁对象1
    private static Object obj2 = new Object();//锁对象2

    //调用传输过来的走向标记
    public DeadLockRunnable(int flag) {
   
        this.flag = flag;
    }

    @Override
    public void run() {
   
        
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值