如何理解Condition

在jdk1.5以后并发包中提供了Lock接口,Condition接口与Lock配合使用可以实现等待/通知模式,在此之前是使用定义在Object对象上的一组监视器方法,主要包括:wait()、wait(long timeout)、notify()以及notifyAll()方法,这些方法synchronized结合使用,也可以实现等待/通知。

Object的监视器方法与Condition接口的对比如下(图片截取自Java并发编程的艺术)

Condition Demo

public class ConditionTest {

        Lock lock =newReentrantLock();

        Condition condition = lock.newCondition();

        public void conditionWait()t hrows InterruptedException{

                    lock.lock();

                    try{

                        System.out.println(Thread.currentThread());

                        condition.await();

                        System.out.println("await");

                        }finally{

                            lock.unlock();

                    }

        }

        public    void    conditionSignal()    throws    InterruptedException{

                lock.lock();

                try{

                    System.out.println(Thread.currentThread());

                    condition.signal();

                    System.out.println("siganl");

                }finally{

                        lock.unlock();

                }

        }

        public    static    void    main( String[] args)    throws    InterruptedException{

                    ConditionTest conditionTest =newConditionTest();

                    newThread(newRunnable() {

                    @Override

                    public    void    run(){

                        try{

                            conditionTest.conditionWait();

                        }catch(InterruptedException e) {

                            e.printStackTrace();

                    }

                }

        }) {

                public    String    toString(){

                returngetName();

            }

        }.start();

            newThread(newRunnable() {

            @Override

                public    void    run(){

                    try{

                        conditionTest.conditionSignal();

                        }catch(InterruptedException e) {

                            e.printStackTrace();

                        }

                   }

            }) {

            public     String    toString(){

            returngetName();

            }

    }.start();

}

}


运行结果:

从上面的demo比较容易得出:

1、一般都会将Condition对象作为成员变量。

2、Thread-1调用await()方法后当前线程会释放锁并等待。

3、线程Thread-1调用signal()方法通知Thread-0,Thread-0从await()返回,在返回前已经获取到了锁。

上面demo中lock.newCondition() 其实返回的是Condition的一个实现:AQS中的ConditionObject

一个ConditionObject包含一个等待队列,ConditionObject包括首节点和尾节点,等待队列是一个FIFO队列,如果一个线程调用Condition.await()方法,那么该线程将会释放锁,构造成节点加入等待队列并进入等待状态,并将该节点从尾部加入等待队列。节点的定义复用了同步器中节点的定义,也就是说,同步队列和等待队列中节点类型都是同步器的静态内部类AbstractQueuedSynchronizer.Node。

等待队列的基本结构如图,同步队列的结构图可以参见AQS简介与源码剖析

Condition接口提供了如下方法:

等待

调用Condition的await()方法(或者以await开头的方法),会使当前线程进入等待队列并释放锁,同时线程状态变为等待状态。当从await()方法返回时,当前线程一定获取了Condition相关联的锁。

如果从队列(同步队列和等待队列)的角度看await()方法,当调用await()方法时,相当于同步队列的首节点(获取了锁的节点)移动到Condition的等待队列中。

通知

调用Condition的signal()方法,将会唤醒在等待队列中等待时间最长的节点(首节点),在唤醒节点之前,会将节点移到同步队列中。

总结

Demo中的具体流程如下:

1、Thread-1调用lock.lock()获取到锁,内部调用acquireQueued方法,线程被加入到AQS的同步队列中。

2、Thread-1调用await方法时,锁释放,该线程锁构成的节点从AQS的同步队列中移除,通过addConditionWaiter()方法加入到Condition的等待队列中,等待着被通知的信号。

3、Thread-1释放锁唤醒tread-2获取到锁,加入到AQS的同步队列中,处理业务

4、Tread-2调用signal方法,Condition的等待队列中只有Thread-1一个节点,通过调用enq(Node node)方法加入到AQS的同步队列中,此时Thread-1 并没有被唤醒,唤醒的操作是在finall块中的lock.unlock()中。

5、Tread-2调用lock.unLock()方法,释放锁,Thread-1被唤醒并获取到锁从await()方法返回继续处理业务。

6、Thread-1调用unlock释放锁,结束整个流程。

参考文章:

Doug Lea:《Java并发编程实战》

方腾飞、魏鹏、程晓明:《并发编程的艺术》


                                        欢迎关注微信公众号获取更多学习资源

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 193,968评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,682评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,254评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,074评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,964评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,055评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,484评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,170评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,433评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,512评论 2 308
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,296评论 1 325
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,184评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,545评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,150评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,437评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,630评论 2 335

推荐阅读更多精彩内容