LockSupport的park和unpark操作

以前见到过很多次这个类—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底层并发库使用的并发控制手段之一,学习该工具对于理解底层库也非常有用,也扩充了自己的并发工具。

你可能感兴趣的:(Java多线程)