阻塞的三种状态以及死锁

2 篇文章 0 订阅

1、一个线程占有了一个对象锁之后,CPU资源仍然能够被抢走
2、当其他线程抢到了CPU资源之后,发现获取不到对象锁,就会进入阻塞状态,此时进入的是由于获取不到对象锁而阻塞的阻塞队列中
3、当占有锁的线程释放锁的时候,就会立即唤醒等待锁的其他线程(在因获取不到对象锁而阻塞的阻塞队列中的线程)。

上面所说的释放锁的时候,包括了所有释放锁的情形
a、线程退出同步块

synchronized(obj){//当进入同步代码块的时候,会尝试获取对象锁
}执行完同步块,直接释放对象锁,并唤醒在阻塞队列中等待的线程

b、线程调用了wait()方法,会释放CPU和锁。因为wait()会引起锁的释放,而一旦释放锁,就会立即唤醒那么等待锁的线程。

线程阻塞的原因有三种
1、尝试获取锁,但是没有得到,就进入了等待锁的阻塞队列中
notify不会唤醒该队列中的线程
2、正在占有锁的线程,调用了wait()方法,就进入了wait()阻塞队列中
只有notify才可以唤醒该队列中的线程
3、正在占有锁的线程,调用sleep()方法或者IO(等待键盘输入),就进入了另一个阻塞队列中。
睡眠时间到或者IO阻塞结束,线程才得以进入可运行状态(就绪状态)

死锁

根本原因:线程有一个锁,还想要另外一个锁。所以必须是两把锁以上才会产生死锁。
当只有一个锁的时候,线程获取了这个锁,然后再次获取这个锁,不会造成死锁。这就是可重入锁。

class T implements Runnable{
    public Object obj1,obj2;

    public T(Object obj1, Object obj2) {
        this.obj1 = obj1;
        this.obj2 = obj2;
    }

    @Override
    public void run() {
        while(true){
            synchronized (obj1){//只有在进入到这里的时候,
                // 刚好T2抢到了obj2对象锁,
                //此时会进入死锁状态,这是一个几率问题
                //所以会先循环输出一段时间,然后出现死锁
                synchronized (obj2){
                    System.out.println("AAAAA");
                    System.out.println("BBBBB");
                }
            }
        }

    }
}

class T2 implements Runnable{
    public Object obj1,obj2;

    public T2(Object obj1, Object obj2) {
        this.obj1 = obj1;
        this.obj2 = obj2;
    }

    @Override
    public void run() {
        while(true){
            synchronized (obj2){
                synchronized (obj1){
                    System.out.println("11111");
                    System.out.println("22222");
                }
            }
        }

    }
}

public class Test2 {
    public static void main(String[] args) {
        Object obj1 = new Object();
        Object obj2 = new Object();
        Thread t1 = new Thread(new T(obj1,obj2));
        Thread t2 = new Thread(new T2(obj1,obj2));

        t1.start();
        t2.start();
    }
}
  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值