并发编程之LockSupport源码分析及简单使用

曾经面试被问到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);
	}
}

测试输出结果如下:

结果分析:

  1. LockSupport不需要在同步代码块里。所以线程间也不需要维护一个共享的同步对象了,从而降低了使用时复杂度。
  2. unpark函数可以优先于park调用,所以不需要担心线程间的执行先后顺序。

以上代码仅供参考,如有不当之处,欢迎指出!!!

更多干货,欢迎大家关注和联系我。期待和大家一起更好的交流、探讨技术!!!

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值