深入理解并发编程-阻塞线程状态

因为看到了LockSupport能够使线程进入阻塞状态,就想了一下还有什么方法能够使线程进入阻塞状态吗?因为我记忆里只有拿不到锁的时候才会进入阻塞状态
搜了一下引发线程进入阻塞的方法
网上有这样说:
(1)线程睡眠:Thread.sleep (long millis)方法,使线程转到阻塞状态。

(2)线程等待:Object类中的wait()方法,

(3)线程礼让,Thread.yield() 方法

(4)线程自闭,join()方法,

(5)suspend() 和 resume() 方法
以上方法可能使阻塞线程,但我认为只有进入了阻塞状态才能被认为是阻塞了线程,可能是我太钻牛角尖了,我觉得这样的说法是有点偏颇的,因为我脑海里记忆的是下面的图:
在这里插入图片描述
在这里插入图片描述
阻塞(Blocked):线程被阻塞了,“阻塞状态”与“等待状态”的区别是“阻塞状态”在等待着获取到一个排它锁,这个事件将在另外一个线程放弃这个锁的时候发生;而“等待状态”则是在等待一段时间,或者唤醒动作的发生。在程序等待进入同步区域的时候,线程将进入这种状态。

但又有没有可能是书上错了呢?
那就来试试吧

1.sleep()方法

 public static void main(String[] args) {
        new Thread(()->{
            try {
                System.out.println("正在睡眠");
                Thread.sleep(100000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        },"t1").start();
    }

使用jps查看线程
在这里插入图片描述
使用jstack查看线程状态:
在这里插入图片描述
哦豁,TimeWaiting,线程处于超时等待状态

2. wait 方法

在这里插入图片描述

    Object o = new Object();
        Thread t2 = new Thread(() -> {
            synchronized (o) {
                try {
                    System.out.println("执行等待!");
                    o.wait();
                } catch (InterruptedException e) {

                }
            }
        }, "t2");
        t2.start();

        Thread.sleep(100000);
        o.notify();

在这里插入图片描述
waitting状态

3.synchronized(出现阻塞状态)

 Object o = new Object();
        new Thread(() -> {
            try {
                Thread.sleep(10);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            synchronized (o) {
            }
        }, "t1").start();
        //让t2 先拿到锁,然后就一直处于wait状态,我们再来观察t1
        new Thread(() -> {
            synchronized (o) {
                try {
                    Thread.sleep(100000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, "t2").start();

在这里插入图片描述

4. join

  Thread t1 = new Thread(() -> {
            try {
                Thread.sleep(10000000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }, "t1");
        t1.start();
        //先让t1 跑起来 观察t2
        Thread.sleep(10);
        Thread t2 = new Thread(() -> {
            try {
                t1.join();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

        }, "t2");
        t2.start();

在这里插入图片描述

5.yield

让出CPU时间片的方法,有点不好操作,因为让出一个时间片的太短了啊

以后想到方法了再来更新

6.LockSupport

在这里插入图片描述

    Thread t1 = new Thread(() -> {
            LockSupport.park();
        }, "t1");
        t1.start();
        Thread.sleep(100000);
        LockSupport.unpark(t1);

在这里插入图片描述

结论:
最后一个LockSupport就把之前的所有可以推翻:只有进入阻塞状态才能称为阻塞线程

可能这一部分定义得不是特别明确,就不深究了,了解就行了,日常开发应该也用不到这一部分的知识,理解就行;

还有如果要阻塞线程:
以上除了yield 其他的都可以起到阻塞线程的作用,而不是让线程进入java定义的阻塞线程状态

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值