LockSupport原理与线程挂起/唤醒

本文一些理解和代码参考了看过的网上一些文章,感谢原作者们

之前在Java并发编程中的等待/通知范式 - 肥兔子爱豆畜子 - 博客园 (cnblogs.com)
中讨论了java并发编程里的“等待-通知”范式,里边提到了LockSupport,最近也在研究之前的一份tomcat线程dump的样本和AQS、也都涉及到这个类,所以这里有必要再深入一下。

LockSupport工具类

LockSupport打开源码一看都是static方法,典型的工具类模式,是一个在jdk里边相对偏底层的工具类,是用来编写其他稍微上层一些的并发库的工具。
作用是用来对Thread进行WAITING(TIMED_WAITING)和RUNNABLE之间的状态转换。对比一下Object.wait和notify先要synchronized锁住object -> wait阻塞等待 -> 释放锁 -> notify -> 抢锁获得锁 -> 从wait往下执行 这样的一个流程,LockSupport不需要先获取对象的锁,可以直接LockSupport.park()阻塞当前线程,LockSupport.park(object)当前线程的阻塞和object锁没有必然关系,仅是用来标识当前线程是阻塞在object上这样一种逻辑关系。

public static void park(Object blocker) {
    Thread t = Thread.currentThread();
    setBlocker(t, blocker); // UNSAFE.putObject(Thread.currentThread(), parkBlockerOffset, blocker);
    UNSAFE.park(false, 0L);
    setBlocker(t, null);
}

可以用如下例子来理解验证一下:

import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.LockSupport;

import lombok.extern.slf4j.Slf4j;

@Slf4j
public class LockSupportTest {
    private static Object lock = new Object();

    static class MyThread extends Thread {

        public MyThread(String name) {
            super(name);
        }

        @Override
        public void run() {
            //synchronized(lock) {
            //System.out.println(this.getName()+"拿到lock并开始执行run");
            log.info(this.getName() + "开始执行");
            LockSupport.park(lock);
            if (this.isInterrupted()) {
                log.info(this.getName() + "被中断");
            }
            log.info(this.getName() + "继续执行");
            //}
        }
    }

    public static void main(String[] args) throws InterruptedException {
        MyThread t1 = new MyThread("t1");
        MyThread t2 = new MyThread("t2");

        t1.start();
        TimeUnit.SECONDS.sleep(1);
        t2.start();
        TimeUnit.SECONDS.sleep(1);

        LockSupport.unpark(t2);
        TimeUnit.SECONDS.sleep(1);
        t1.interrupt();

        t1.join();
        t2.join();
    }
}

执行结果:

14:11:45.617 [t1] INFO com.lock.test.LockSupportTest - t1开始执行
14:11:46.620 [t2] INFO com.lock.test.LockSupportTest - t2开始执行
14:11:47.623 [t2] INFO com.lock.test.LockSupportTest - t2继续执行
14:11:48.626 [t1] INFO com.lock.test.LockSupportTest - t1被中断
14:11:48.626 [t1] INFO com.lock.test.LockSupportTest - t1继续执行

45秒t1开始执行,然后阻塞在LockSupport.park(lock),1秒后t2开始执行,然后也是阻塞在LockSupport.park(lock),1秒后unpark(t2),t2继续执行,再1秒后,t1.interrupt()然后t1继续执行。这是完整的一个流程。可见park(lock)跟object锁没关系。

LockSupport比较偏底层,再下面就是unsafe和native方法了,总之是通过修改内存中线程的一个布尔类型的permit字段、来标识该线程是否可被操作系统进行调度:true则代表可以被调度、false则系统不会调度该线程即挂起。后面会进一步分析permit的作用。
至于更为底层的java对象在内存中的layout以及本地线程调度的知识,这里先不表,点到为止。。。

image

LockSupport源码注释分析

在已经了解了LockSupport里边的park(), parkNanos(), unpark()这些基本方法的用法以后,其实更进一步深入可以看源代码和上边的注释,比如park()方法:

public static void park() {
    UNSAFE.park(false, 0L);
}

源代码过于简单,看下注释:

If the permit is available then it is consumed and the callreturns immediately;
otherwise the current thread becomes disabled for thread scheduling purposes and lies dormant until one of threethings happens:
•Some other thread invokes unpark with thecurrent thread as the target; or
•Some other thread interruptsthe current thread; or
•The call spuriously (that is, for no reason) returns.
This method does not report which of these caused themethod to return. Callers should re-check the conditions which causedthe thread to park in the first place. Callers may also determine,for example, the interrupt status of the thread upon return.
This method does not report which of these caused the method to return. Callers should re-check the conditions which caused the thread to park in the first place. Callers may also determine,for example, the interrupt status of the thread upon return.

如果permit是可用(permit=true),那么使用消费掉permit并返回。否则permit=false线程不被调度进入休眠,直到

  1. 被其他线程用unpark()唤醒
  2. 其他线程发送中断信号给当前线程,thread.interrupt()
  3. 调用不合逻辑的(毫无理由的)返回
    这个方法不报告上述这些返回的原因,调用者需要重新检查最先导致线程park的条件。调用者也可以在返回后确认线程的中断状态。

再看看unpark()

public static void unpark(Thread thread) {
    if (thread != null)
        UNSAFE.unpark(thread);
}

Makes available the permit for the given thread, if it was not already available.
If the thread was blocked on park then it will unblock. Otherwise, its next call to park is guaranteed not to block.
This operationis not guaranteed to have any effect at all if the giventhread has not been started.

如果当前线程的permit==false是不可用的,那么将permit变为可用,permit=true。
解除当前线程在park上的阻塞,或者如果当前线程没阻塞在park上、那么将会保证其下一次park不会被阻塞。
如果线程还没start,则不会得到上面的保证。

总结

综合上面信息,我们可以知道:

  • park是响应中断的,中断会使得park结束,只不过不会抛中断异常,可以在park返回时检查中断状态来确认被中断了。
  • 然后park的时候是要检查内部的一个线程permit状态的(对调用者透明),如果permit=true,那么park就不阻塞,直接返回,如果permit=false则阻塞。
  • unpark相当于修改thread对应的permit为true,如果此时线程刚好处于park阻塞,则马上放行,如果没在park则由于permit=true,该线程的下一次park不会被阻塞。这个特性使得即使unpark先于park调用也不会导致notify先于wait那样发生死锁。
  • permit=true用过以后就消费掉了,变成permit=false
  • LockSupport.park阻塞线程不用先拿object锁(可以用park(object)来标识逻辑关系),它是用底层native方法去修改线程上的一个Permit状态控制线程调度系统是否调度这个线程、来达到阻塞还是执行,所以其用法比Object.wait()/notify()更灵活方便。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,165评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,720评论 1 298
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,849评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,245评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,596评论 3 288
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,747评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,977评论 2 315
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,708评论 0 204
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,448评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,657评论 2 249
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,141评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,493评论 3 258
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,153评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,108评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,890评论 0 198
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,799评论 2 277
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,685评论 2 272

推荐阅读更多精彩内容