曾经面试被问到JAVA中实现线程挂起和唤醒有哪些方式,除了采用Object类中wait notify/notifyAll实现还有其他方式吗?当时真心没答上来,后来做了功课才知道还有LockSupport类中park/unpark可以实现。
大家如果了解AQS(全称AbstractQueuedSynchronizer),就知道并发编程中很多类都是基于AQS实现的,例如ReentrantLock、 CountDownLatch、Semaphore等等。
通过分析AQS源码,不难发现AbstractQueuedSynchronizer中ConditionObject内部类就采用LockSupport实现线程挂起和唤醒。
例如ConditionObject中await方法如下:
例如ConditionObject中signal方法如下:
ConditionObject中signal方法调用doSignal方法如下:
ConditionObject中doSignal调用外部类的transferForSignal方法如下:
我们可以看出LockSupport类中park/unpark同样可以实现线程挂起和唤醒。其实如果大家看过我之前写的‘手写一个同步类容器(互联网大厂面试题目)’文章,就知道文章有采用Lock/Condition await signalAll实现 更精确定位唤起的线程,它的底层实现就是LockSupport的park/unpark。
我们来分析LockSupport源码:
通过LockSupport源码设计我们知道它实现线程挂起和唤醒是基于Unsafe底层实现。代码不是很复杂,我们这里不做详细
讲解。
我们来找到Unsafe中park、unpark源码:
由此,我们就完整的跟进了LockSupport类中park/unpark的设计理念。
现在我们通过一段简单的代码,来初步理解LockSupport如何挂起和唤醒线程。
package com.locksupport;
import java.util.concurrent.locks.LockSupport;
/**
* LockSupport源码分析、使用场景说明
*
* @author 小辉GE/小辉哥
* <p>
* 2019年8月30日 下午21:30:00
*/
public class LockSupportAnalysis {
public static void main(String[] args) throws Exception {
Thread thread = new Thread(() -> {
System.out.println("begin......");
// 当前线程wait
LockSupport.park();
System.out.println("end......");
});
thread.start();
// Thread.sleep(1000);
// 线程解除wait
LockSupport.unpark(thread);
}
}
测试输出结果如下:
结果分析:
- LockSupport不需要在同步代码块里。所以线程间也不需要维护一个共享的同步对象了,从而降低了使用时复杂度。
- unpark函数可以优先于park调用,所以不需要担心线程间的执行先后顺序。
以上代码仅供参考,如有不当之处,欢迎指出!!!
更多干货,欢迎大家关注和联系我。期待和大家一起更好的交流、探讨技术!!!