温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

线程阻塞唤醒工具LockSupport怎么使用

发布时间:2023-01-28 15:38:02 来源:亿速云 阅读:149 作者:iii 栏目:开发技术

本篇内容主要讲解“线程阻塞唤醒工具LockSupport怎么使用”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“线程阻塞唤醒工具LockSupport怎么使用”吧!

    LockSupport 简介

    LockSupport 是 Java 并发编程中一个非常重要的组件,我们熟知的并发组件 Lock、线程池、CountDownLatch 等都是基于 AQS 实现的,而 AQS 内部控制线程阻塞和唤醒又是通过 LockSupport 来实现的。

    从该类的注释上也可以发现,它是一个控制线程阻塞和唤醒的工具,与以往的不同是它解决了曾经 wait()、notify()、await()、signal() 的局限。

    回顾 synchronized 和 Lock

    我们知道 Java 中实现并发安全通常会通过这两种加锁的方式,对于 synchronized 加锁的方式,如果我们想要控制线程的阻塞和唤醒是通过锁对象的 wait()notify() 方法,以下面循环交替打印 AB 为例

    int status = 2;
    public static void main(String[] args) throws InterruptedException {
        TestSync obj = new TestSync();
         new Thread(() -> {
            synchronized (obj){
                while (true){
                    if(obj.status == 1){
                        obj.wait();
                    }
                    System.out.println("A");
                    obj.status = 1;
                    TimeUnit.SECONDS.sleep(1);
                    obj.notify();
                }
            }
         }).start();
        new Thread(() -> {
           synchronized (obj){
              while (true){
                  if(obj.status == 2){
                      obj.wait();
                  }
                  System.out.println("B");
                  obj.status = 2;
                  TimeUnit.SECONDS.sleep(1);
                  obj.notify();
              }
           }
        }).start();
    }

    如果我们使用 Lock 实现类,上述代码几乎是一样的,只是先获取 Condition 对象

     Condition condition = lock.newCondition();

    obj.wait() 换成 condition.await()obj.notify() 换成 condition.signal() 即可。

    LockSupport 和 synchronized 和 Lock 的阻塞方式对比

    技术阻塞唤醒方式局限
    synchronized使用锁对象的 wait()、notify()1. 只能用在 synchronized 包裹的同步代码块中 2. 必须先 wait() 才能 notify()
    Lock使用 condition 的 await()、signal()1. 只能用在 lock 锁住的代码块中 2. 必须先 await() 才能 signal()
    LockSupportpark()、unpark(Thread t)没有限制

    LockSupport 的使用

    下面代码中,我们使用 LockSupport 去阻塞和唤醒线程,我们可以多次尝试,LockSupportpark()unpark() 方法没有先后顺序的限制,也不需要捕获异常,也没有限制要在什么代码块中才能使用。

        public static void main(String[] args) throws InterruptedException {
            Thread t1 = new Thread(() -> {
                System.out.println("A");
                LockSupport.park();
                System.out.println("被唤醒");
            });
            t1.start();
            TimeUnit.SECONDS.sleep(2);
            new Thread(() -> {
                System.out.println("B");
                LockSupport.unpark(t1);
            }).start();
        }

    LockSupport 注意事项

    许可证提前发放

    从该类的注释中我们可以看到这个类存储了使用它的线程的一个许可证,当调用 park() 方法的时候会判断当前线程的许可证是否存在,如果存在将直接放行,否则就阻塞。

    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(() -> {
            try {
                TimeUnit.SECONDS.sleep(3);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            System.out.println("A");
            LockSupport.park();//不会阻塞
            System.out.println("被唤醒");
        });
        t1.start();
        TimeUnit.SECONDS.sleep(2);
        new Thread(() -> {
            System.out.println("B");
            System.out.println("先调用 unpark()");
            LockSupport.unpark(t1);
        },"t2").start();
    }

    看这个代码示例,这里我们在 t2 中先让线程 t1 unpark(), 然后在 t1 中调用 park(), 结果并不会阻塞 t1 线程。因为在 t2 中调用 LockSupport.unpark(t1); 的时候相当于给 t1 提前准备好了许可证。

    许可证不会累计

    LockSupport.unpark(t1); 无论调用多少次,t1 的通行证只有一个,当在 t1 中调用两次 park() 方法时线程依然会被阻塞。

    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(() -> {
            try {
                TimeUnit.SECONDS.sleep(3);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            System.out.println("A");
            LockSupport.park();
            LockSupport.park();
            System.out.println("被唤醒");
        });
        t1.start();
        TimeUnit.SECONDS.sleep(2);
        new Thread(() -> {
            System.out.println("B");
            System.out.println("先调用 unpark()");
            LockSupport.unpark(t1);
            LockSupport.unpark(t1);
            LockSupport.unpark(t1);
            LockSupport.unpark(t1);
            LockSupport.unpark(t1);
        },"t2").start();
    }

    以上述代码为例,t1 将被阻塞。

    LockSupport 底层实现

    观察源码发现 park() 和 unpark() 最底下调用的是 native() 方法,源码在 C++ 中实现

    @IntrinsicCandidate
    public native void park(boolean isAbsolute, long time);
    @IntrinsicCandidate
    public native void unpark(Object thread);

    到此,相信大家对“线程阻塞唤醒工具LockSupport怎么使用”有了更深的了解,不妨来实际操作一番吧!这里是亿速云网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

    向AI问一下细节

    免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

    AI