前文我们介绍了wait()和notify()方法,而park和Unpark在一些用法上与二者相似
一、简单介绍
他们是LockSupport类中的方法
用法:
public static void main(String[] args) {
Thread t1 = new Thread(()->{
System.out.println("start");
try {
Thread.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("park");
LockSupport.park();//暂停当前线程
System.out.println("resume");
});
t1.setName("t1");
t1.start();
try {
Thread.sleep(2);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("Unpark");
LockSupport.unpark(t1);//唤醒该线程
}
二、原理
每个线程都有自己的一个 Parker 对象,由三部分组成 counter , cond 和 mutex 打个比喻。
线程就像一个旅人, Parker 就像他随身携带的背包,条件变量就好比背包中的帐篷。 counter 就好比背包中的备用干粮(0为耗尽,1为充足),调用 park 就是要看需不需要停下来歇息。
1、如果备用干粮耗尽,那么钻进帐篷歇息
2、如果备用干粮充足,那么不需停留,继续前进
调用 unpark ,就好比令干粮充足
1、如果这时线程还在帐篷,就唤醒让他继续前进
2、如果这时线程还在运行,那么下次他调用 park 时,仅是消耗掉备用干粮,不需停留继续前进
因为背包空间有限,多次调用 unpark 仅会补充一份备用干粮
三、与Object的wait和notify进行比较
1、wait , notify 和 notifyAl 必须配合 Object Monitor 一起使用,而 unpark 不必
2、park & unpark 是以线程为单位来【阻塞】和【唤醒】线程,而 notify 只能随机唤醒一个等待线程
3、notifyAl 是唤醒所有等待线程,就不那么【精确】
4、park & unpark 可以先 unpark ,而 wait & notify 不能先 notify