Java ReentrantLock中condition通信的好处

之前看到Java Concurrent包中有个Condition接口。这个接口如今已经普遍用于线程通信, 使用方法主要依靠condition的await方法和signal方法,但这一对方法和Java经典的wait,notify方法对颇为相似。但这个新的方法对有什么好处呢,思考过后得出一句结论:减少无谓的唤醒。
于是写下这篇文章做个简单的笔记,文章首先简要介绍一下预备知识,但不打算详细说,毕竟重点仅放在condition上。介绍完毕后就是Condition的使用方法以及举例说明Condition好处在哪。

Java线程间的通信

Java 线程通信最常用的就是经典的三种:

  1. volatile 共享变量轮询
  2. synchronized 下使用object的wait,notify方法对
  3. ReentrantLock下使用condition的await,signal方法对

volatile 共享变量轮询, 核心代码如下

public class Main{
    volatile boolean shouldStop = false;
    Thread thread_1 = new Thread(){
           @Override
           public void run() {
               while(!shouldStop){
                   //do something
               }
           }
       };
       Thread thread_2 = new Thread(){
           @Override
           public void run() {
               try{
                   sleep(1000);
                   shouldStop = true;
               }catch (InterruptedException e){
                   e.printStackTrace();
               }
           }
        };
}

线程1和线程2通过共享shouldStop来决定是否停止工作,至于为什么要用volatile关键字,主要有两点:

  1. 强制共享变量修改时flush回主存
  2. 禁止cpu优化代码时的指令重排
    具体的可以看这里 http://www.importnew.com/23535.html

synchronized中使用wait,notify方法对
虽然这个方法估计各位大佬都已经熟烂了,但为了和await,signal机制做对比,请允许我写一个生产者/消费者 模型来做说明。

public interface Buffer {
    void put(Integer integer) throws InterruptedException;
    Integer take() throws InterruptedException;
}
import java.util.ArrayList;

public class ClassicBuffer implements Buffer{
    private Object lock = new Object();
    private final static int CAPACITY = 1;
    private int count = 0;
    private ArrayList<Integer> list = new ArrayList<>(CAPACITY);
    public ArrayList<Integer> getList(){
        return list;
    }
    public  void put(Integer e) throws InterruptedException{
        if(e == null){
            return;
        }
        synchronized (lock) {
    
            try{
                while(count == CAPACITY){
                    lock.wait();
                    System.out.println("Classic_Put: "+Thread.currentThread());
                }
                list.add(e);
                count++;
                lock.notifyAll();
            }catch (InterruptedException exception) {
                // TODO: handle exception
                exception.printStackTrace();
            }
        }
    }
    public synchronized Integer take() throws InterruptedException{
    
        synchronized (lock) {
            
            Integer e = -1;
            try{
                while(count == 0){
                    lock.wait();
                    System.out.println("Classic_Take: "+Thread.currentThread() );
                }
                 e = list.get(count % CAPACITY);
                count --;
                lock.notifyAll();
                return e;
            }catch (InterruptedException exception) {
                // TODO: handle exception
                return e;
            }
        }
    }
}

这是用object的notify和wait来实现阻塞队列的核心代码,稍微解释一下代码含义。
阻塞队列实现Buffer接口,这个接口只有put和take两个方法, 容量大小为定义好的常量CAPACITY,这里是1,当前容量用count变量来统计。

生产者(put):
put的时候如果满足当前容量count 等于容量CAPACITY,那说明队列已经满了,不能再投放数据了,因此要用wait()来阻塞自己。如果容量未满,那么可以投放数据,一旦投放数据,队列就不为空,此时很有可能有一些消费者在阻塞等待队列不为空,因此这时候要唤醒这些等待的消费者。这里用的是notifyAll来做唤醒(个人觉得不应该使用notify,因为notify只会随机唤醒一条线程,如果有多条生产者线程会出现麻烦,后面会细细道来)。

消费者(take):
逻辑和生产者相似,如果当前容量count已经等于0,那么说明队列为空,没有数据,因此消费者需要wait自己来阻塞等待数据到来。如果容量不为空,那么消费者会取走一个数据,容量减少,因此队列此时一定不满,需要notifyAll来唤醒阻塞中的生产者。

另外: 生产者消费者只需要在runnable中实现调用这个阻塞队列的put/take就可以了,这部分的代码会在本文末章奉上。

ReentrantLock中使用condition 的await,signal方法对

import java.util.ArrayList;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

public class ConditionBuffer implements Buffer{
    private Lock lock = new ReentrantLock();
    private Condition notEmpty = lock.newCondition();
    private Condition notFull = lock.newCondition();
    private final static int CAPACITY = 1;
    private int count = 0;
    private ArrayList<Integer> list = new ArrayList<>(CAPACITY);
    public ArrayList<Integer> getList(){
        return list;
    }
    public void put(Integer e) throws InterruptedException{
        if(e == null){
            return;
        }
        lock.lock();
        try{
            while(count == CAPACITY){
                notFull.await();
                System.out.println("Reentrant_put: "+Thread.currentThread());
            }
            list.add(e);
            count++;
            notEmpty.signal();
        }finally {
            lock.unlock();
        }
    }
    public Integer take() throws InterruptedException{
        lock.lock();
        while(count == 0){
            notEmpty.await();
            System.out.println("Reentrant_take: "+Thread.currentThread());
        }
        try{
            Integer e = list.get(count % CAPACITY);
            count --;
            notFull.signal();
            return e;
        }finally {
            lock.unlock();
        }
    }
}

这段代码的逻辑和上一段是一样的,不同的地方是使用ReentrantLock代替synchronized来做同步, 用condition代替object来做线程通信。
具体的使用方法跟object的wait,notify很相似,await和signal同样要在同步区中调用,并且使用ReentrantLock要记得手动unlock。稍微提一提ReentrantLock。

ReentrantLock是 Java concurrent包里实现的可重入锁机制。它和synchronized的主要区别是
ReentrantLock是在java层面上实现的,基于AQS(AbstractQueuedSynchronized)框架下使用自旋CAS机制实现,另外ReentrantLock扩展了很多额外的同步方法,比如公平锁,非公平锁,可中断锁,非阻塞锁。
而synchronized是基于JVM层面实现的,使用计数监视锁来做同步。
具体可以到这里看 http://hanhailong.com/

Condition比object通信好在哪

扯了那么多,终于来到做笔记的地方啦。再次说一遍好处:condition减少无谓的唤醒。
咱们现在开始把生产消费搞起,做一次测试。
生产者线程:

public class Producer implements Runnable{
    Buffer buffer;
    public Producer(Buffer buffer){
        this.buffer = buffer;
    }
    public void run(){
        try{
            while(true){
                buffer.put(1);
            }
        }catch (InterruptedException e) {
            e.printStackTrace();
            // TODO: handle exception
        }
    }
}

消费者线程

public class Consumer implements Runnable {
    Buffer buffer;
    public Consumer(Buffer buffer){
        this.buffer = buffer;
    }
    public void run(){
        try{
            while(true){
                buffer.take();
            }
        }catch (InterruptedException e) {
            e.printStackTrace();
            // TODO: handle exception
        }
    }
}

很简单对吧,仅仅是把实现好的阻塞队列注入到线程中。好,现在我们创建三条生产者线程,一条消费者线程。走起

public class Main {
    public static void main(String[] args){
        ClassicBuffer classicBuffer = new ClassicBuffer();
        ConditionBuffer blockBuffer = new ConditionBuffer();
        Thread thread_1;
        Thread thread_2;
        Thread thread_3;
        Thread thread_4;
        Consumer consumer;
        Producer producer;
        if(args[0].contains("classic")){
            consumer = new Consumer(classicBuffer);
            producer = new Producer(classicBuffer);
        }
        else{
            consumer = new Consumer(blockBuffer);
            producer = new Producer(blockBuffer);
        }
        thread_1 = new Thread(consumer);
        thread_2 = new Thread(producer);
        thread_3 = new Thread(producer);
        thread_4 = new Thread(producer);
        
        thread_1.start();
        thread_2.start();
        thread_3.start();
        thread_4.start();
    }
}

0号是来看看结果吧,先上condition的结果

Condition_reentrant_result.png

因为队列只有1容量,出现了与预想中一样很均匀的线程切换: 一个生产者,一个消费者轮流切换,没有任何多余的线程唤醒。

再看object wait/notify的结果

Condition_synchronized_result.png

是时候做分析了
我们先看回上面的object和condition实现的阻塞队列代码。再次贴一些关键的部分, 以生产者为例,

// ConditionBuffer.Put()
lock.lock();
try{
    while(count == CAPACITY){
        notFull.await();
        System.out.println("Reentrant_put: "+Thread.currentThread());
    }
    list.add(e);
    count++;
    notEmpty.signal();
}finally {
    lock.unlock();
}

//ClassicBuffer.Put()
synchronized (lock) {   
try{
    while(count == CAPACITY){
        lock.wait();
        System.out.println("Classic_Put: "+Thread.currentThread());
    }
    list.add(e);
    count++;
    lock.notifyAll();
}catch (InterruptedException exception) {
// TODO: handle exception
    exception.printStackTrace();
    }
}

很明显,对比两个结果,object实现的结果比condition实现的结果每次多了两条无谓线程的切换,因为object每次是以notifyAll来唤醒的,所以所有等待中的线程,无论是生产者和消费者都要被唤醒。
但考虑到队列容量只有1,当生产者线程1完成数据插入时,它会把生产者线程2,3以及消费者线程0给唤醒,显然,生产者线程此时被唤醒之后做的唯一一件事就是判断容量是否等于1,由于此时生产者线程1刚刚完成插入,因此,2,3生产者发现容量等于1,再次进入wait,相当于他们这次醒来什么都没干,造成线程切换的浪费。

然而聪明的你们可能已经发现了"你这不公平!凭什么object要用notifyAll,而condition用的是signal并非是signalAll!"
好,好,先把刀放下,signal能够完成任务咱就不讨论用signalAll了,因为有快的方法就没有必要用慢的对吧。那我们讨论能不能用notify,把消费者所有的notifyAll改成notify,代码就不贴出来了,直接看结果。

Condition_notify_result.png

咦?程序卡住不动了。为什么?
我们分析一下结果

  1. 程序刚进入,0号消费者启动: 队列容量0, 发现容量为0,阻塞自己。
  2. 生产者2号启动: 发现队列容量为0,插入数据,容量变为1。notify唤醒别的线程,然而很不幸,它唤醒了1号生产者。
  3. 生产者1号启动:发现队列容量为1,接着睡。
    就这样结束了,再也没有别的线程能唤醒整个系统,因此卡死了。

但是为什么condition只用signal就可以,而不需要用signalAll呢?
因为condition只会唤醒获得相同条件锁的线程。也就是生产者唤醒的永远是消费者, 反之亦然。参考上面代码,生产者使用notEmpty.signal(),而它本身是以notFull.await()来阻塞自己的,所以生产者并不会唤醒生产者,消费者大家可以同样去分析。
好了,大概就是这样,如果有什么不满意的,欢迎讨论。

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

推荐阅读更多精彩内容

  • layout: posttitle: 《Java并发编程的艺术》笔记categories: Javaexcerpt...
    xiaogmail阅读 5,813评论 1 19
  • Java-Review-Note——4.多线程 标签: JavaStudy PS:本来是分开三篇的,后来想想还是整...
    coder_pig阅读 1,643评论 2 17
  • Java并发总结 1.多线程的优点 资源利用率更好 程序在某些情况下更简单 程序响应更快 2.创建线程 1.实现R...
    不会上树的猴子阅读 1,027评论 0 5
  • 日落西山凹,人约黄昏后。 请君多珍重,他日在聚首。
    郝逗阅读 166评论 0 0
  • 雷雁雄8月22日总结:今天早上和合作伙伴在一起再次策划周年店庆的活动,我发现我和合作伙伴都有拖延症,每件事都是逼不...
    雷雁雄阅读 125评论 0 0