park和unpark方法详解

简述

  他们都是LockSupport,park用于暂停某个线程,unpark用于恢复某个线程的运行。

@slf4j
public class Test {
    public static void main(String[] args) {
        Thread t1 = new Thread(() -> {
            log.debug("start...");
            sleep(1);
            log.debug("park...");
            LockSupport.park();
            log.debug("resume...");
        }, "t1");
        t1.start();
        sleep(2);
        log.debug("unpark...");
        LockSupport.unpark(t1);
    }
}

  输出

18:43:50.765 c.TestParkUnpark [t1] - start... 
18:43:51.764 c.TestParkUnpark [main] - unpark... 
18:43:52.769 c.TestParkUnpark [t1] - park... 
18:43:52.769 c.TestParkUnpark [t1] - resume...

特点

  park和unpark一起使用时,可以先unpark,后park的线程停止后会自动唤醒

@slf4j
public class Test {
    public static void main(String[] args) {
        Thread t1 = new Thread(() -> {
            log.debug("start...");
            sleep(2);
            log.debug("park...");
            LockSupport.park();
            log.debug("resume...");
        }, "t1");
        t1.start();
        sleep(1);
        log.debug("unpark...");
        LockSupport.unpark(t1);
    }
}

  输出

18:43:50.765 c.TestParkUnpark [t1] - start... 
18:43:51.764 c.TestParkUnpark [main] - unpark... 
18:43:52.769 c.TestParkUnpark [t1] - park... 
18:43:52.769 c.TestParkUnpark [t1] - resume...

与Object的wait&notify对比

  wait,notify 和 notifyAll 必须配合 Object Monitor 一起使用,而 park,unpark 不必

  park & unpark 是以线程为单位来【阻塞】和【唤醒】线程,而 notify 只能随机唤醒一个等待线程,notifyAll是唤醒所有等待线程,就不那么【精确】

  park & unpark 可以先 unpark,而 wait & notify 不能先 notify

park&unpark原理

  park和unpark会调用Unsafe类中的native方法

  每个线程都会和一个park对象关联起来,由三部分组成 _counter , _cond 和 _mutex。核心部分是counter,我们可以理解为一个标记位。

  当调用park时会看counter是否为0,为0则进入阻塞队列。为1则继续运行并将counter置为0。

  当调用unpark时,会将counter置为1,若之前的counter值为0,还唤醒阻塞的线程。

 

  

  

  

 

posted @ 2021-08-05 08:44  艾尔夏尔-Layton  阅读(665)  评论(0编辑  收藏  举报