以前见到过很多次这个类—java.util.concurrent.locks.LockSupport,然而却没有当一回事。一次偶然的机会看到dump文件中很多park操作,这才仔细的研究了一番。
LockSupport类似于信号量中的二元信号,主要有如下2个方法:
public static void unpark(Thread thread)
释放thread线程的许可证
public static void park()
当前线程获取许可证
注意:这里的许可证也可以理解成锁,或者信号量也行,或者一种资源,只是一种叫法而已。
LockSupport主要有3个特点:
1、默认占有许可证
public static void a1() {
LockSupport.park(); // 许可默认是被占用的,一直在这里等待.
}
如果执行这段代码,线程会一直等待,等待许可证被释放才能继续执行。
2、不可重入
public static void a2() {
Thread thread = Thread.currentThread();
LockSupport.unpark(thread);
LockSupport.unpark(thread); //可以多次释放一个许可证,多次执行也只会释放一个许可证。
LockSupport.park(); //但是一次只能获取一个许可证(不可重入)
LockSupport.park(); //线程在这里等待
System.out.println("--");
}
unpark操作可以执行多次,但是只有一个许可证被释放(线程永远只会有一个)。但是park方法每次调用必须获取一个许可证才能继续执行,因此这段代码会一直等待。
正确的释放和获取方式如下:
public static void a2() {
Thread thread = Thread.currentThread();
LockSupport.unpark(thread);
LockSupport.park();
LockSupport.unpark(thread);
LockSupport.park();
System.out.println("--");
}
注意:许可证可以提前释放,而不必等待获取的时候才释放。
3、可以响应中断
public static void a3() throws Exception {
Thread t = new Thread(new Runnable() {
public void run() {
System.out.println("thread...");
LockSupport.park();
System.out.println("thread done.");
}
});
t.start();
Thread.sleep(2000);
t.interrupt(); //如果因为park而被阻塞,可以响应中断请求,并且不会抛出InterruptedException。
System.out.println("main thread done.");
}
park操作获取许可证可以被中断而继续执行,并且不会抛出异常信息。
总结
LockSupport是许多JDK底层并发库使用的并发控制手段之一,学习该工具对于理解底层库也非常有用,也扩充了自己的并发工具。