Java 数据结构—阻塞队列学习笔记

特点

阻塞队列 BlockingQueue 是线程安全,所有的操作都加了锁。

阻塞你是怎么理解的呢?

队列主要的操作不外乎就是出队和入队两种方式,但是 BlockingQueue对于出队和入队操作做了阻塞操作

简单理解如下

  • 当你要取一个数据时,发现队列为空,那么你就在此等待,等其它人(其它线程)往队列存储存入数据,那它会通知队列有数据了,你就可以取数据了。
  • 当你要存一个数据时,发现队列已满,那么你就在此等待,等其它人(其它线程)从队列中取出数据,那它会通知你队列没有满哦,你就可以存数据了。

如何实现阻塞的呢?

内部主要是通过通过 ReentrantLock 保证每一个操作都是线程安全的,并且通过等待唤醒机制来实现阻塞功能。

下面通过两个基本的存(put)取(take)方法来简单描述一下 BlockingQueue 是如何实现线程安全和阻塞机制的。

  • 入队
public void put(E e) throws InterruptedException {
    Objects.requireNonNull(e);
    final ReentrantLock lock = this.lock;
    //加锁
    lock.lockInterruptibly();
    try {        
        while (count == items.length)
            //队列满了,就等待其它人取走
            notFull.await();
       //队列还没满,可以入队
       enqueue(e);
    } finally {
        //释放锁
        lock.unlock();
    }
}
  • 出队
public E take() throws InterruptedException {
    final ReentrantLock lock = this.lock;
    //还是加锁
    lock.lockInterruptibly();
    try {
        while (count == 0)
            //哇哇,队列为空,等等吧...
            notEmpty.await();
            
        return dequeue();//队列不为空,取一个数据给我
    } finally {
        //还是释放锁
        lock.unlock();
    }
}

阻塞队列主要有三种实现类

SynchrousBlockingQueue

没有容量的队列

ArrayBlokingQueue

底层使用数组实现的阻塞队里,必须要指定队列容量

LinkedBlockingQueue

底层使用链表实现的阻塞队列,可以指定队列容量,默认是 Integer.MAX_VALUE

BlockingQueue 基础 API

  • boolean offer(E e);

往队列中添加一个数据,如果队列还没满,那么返回 true,否则则返回false,表示添加失败,这时需要取出数据之后才能存储数据。该方法在 ThreadPoolExecutor#execute(runnable)中有应用,后面会简单描述。

  • boolean offer(E e, long timeout, TimeUnit unit) throws InterruptedException;

往队列中添加一个数据,如果队列还没满,那么返回 true,如果队列满了,就开始在当前线程阻塞等待,如果等待时间到了,还没有能获取到数据,那么就返回 false

  • void put(E e) throws InterruptedException;

阻塞式添加数据到队列,直到队列可以存放数据。

  • E take() throws InterruptedException;

阻塞式获取数据

  • boolean remove(Object o);

从队列中移除数据

关于上面的 api ,其实我们写个 demo 练习一下就知道它是怎么工作的啦。

不过呢,我对 SynchrousBlockingQueue 是比较有疑问的,它没有容量,也就是不能存储数据,那还能实现阻塞吗?带着疑问,我们来验证一下咯

SynchrousBlockingQueue

  • offer 函数的理解(一)
final SynchronousQueue<String> synchronousQueue = new SynchronousQueue<>();
boolean result = synchronousQueue.offer("1");
System.out.println(result ? "成功添加了第1个数据" : "第1个数据添加失败");
result = synchronousQueue.offer("2");
System.out.println(result ? "成功添加了第2个数据" : "第2个数据添加失败");

因为没有容量啦,所以 offer 结果就直接返回 false了...

第1个数据添加失败
第2个数据添加失败

但是,就开始疑问了,这个有啥用啊?会有返回 true 的情况吗?

  • offer 函数的理解(二)

来看第二个栗子

final SynchronousQueue<String> synchronousQueue = new SynchronousQueue<>();
new Thread() {
    @Override
    public void run() {
        super.run();
        while (true) {
            String s = null;
            try {
                //take() 会一直阻塞
                s = synchronousQueue.take();
                //等待时间超时获取元素
                // s = synchronousQueue.poll(100, TimeUnit.MICROSECONDS);
            } catch (Exception e) {
                e.printStackTrace();
            }
            if (s != null) {
                System.out.println("取出数据:" + s);
            }
        }
    }
}.start();
/*
offer向队列中提交一个元素,如果此时有其他线程正在被take阻塞
(即其他线程已准备接收)或者"碰巧"有poll/take操作,那么将返回true,否则返回false.
 */
Thread.sleep(100);//确保上面的线程跑起来呀
boolean result = synchronousQueue.offer("1");
System.out.println(result ? "成功添加了第1个数据" : "第1个数据添加失败");
Thread.sleep(100);
result = synchronousQueue.offer("2");
System.out.println(result ? "成功添加了第2个数据" : "第2个数据添加失败");

看到了吗?offer(...)操作时,如果此时其它线程正在阻塞等待别人入队,那就刚好这时,我入队了,所以我 offer 返回 true 了啦~

成功添加了第1个数据
取出数据:1
成功添加了第2个数据
取出数据:2

ArrayBlokingQueue

对于 ArrayBlokingQueue 也使用两个函数来看看出入队列的操作吧

  • put 函数的理解
//定义一个只有一个容量的队列,没有实际用处,只是测试而已哈
final ArrayBlockingQueue arrayBlockingQueue = new ArrayBlockingQueue(1);

//添加三个数据
arrayBlockingQueue.put(new Runnable() {
    @Override
    public void run() {
    }
});
System.out.println(System.currentTimeMillis() + " 添加第1个数据");
arrayBlockingQueue.put(new Runnable() {
    @Override
    public void run() {
    }
});
System.out.println(System.currentTimeMillis() + " 添加第2个数据");
//等待中ing...........................我阻塞住了啊,等着别人取走队列的数据,我才能入队呀~
arrayBlockingQueue.put(new Runnable() {
    @Override
    public void run() {
    }
});
System.out.println(System.currentTimeMillis() + " 添加第3个数据");

输出结果:从输出结果我们可以看出,因为 ArrayBlockingQueue 只有一个容量,因此它只能存储一个数据,当第二个数据来时,那它就在那里等待ing了~~

1566057339377 添加第1个数据

那我如果要存储2和3两个数据怎么办?之前说过,阻塞是针对队列是否满来判断的,现在队列已经有一个数据1了,那么当前线程已经阻塞住啦,它的阻塞内部就是调用了 notFull.await() 表示队列满啦,你等着吧

那我找个线程出队列取,然后通过 notFull.signal(); 去通知不就行啦。

来,看看 ArrayBlockingQueueu 在出队之后是怎么通知的?

notFull.signal() 通知等待线程,我有一个数据已经出队了,你可以来存啦。

private E dequeue() {
    // assert lock.getHoldCount() == 1;
    // assert items[takeIndex] != null;
    final Object[] items = this.items;
    @SuppressWarnings("unchecked")
    E x = (E) items[takeIndex];
    items[takeIndex] = null;
    if (++takeIndex == items.length) takeIndex = 0;
    count--;
    if (itrs != null)
        itrs.elementDequeued();
    //这里就是通知当前线程不满,可以存啦~
    notFull.signal();
    return x;
}
  • take 函数的理解

我们来新起一个线程去队列中取数据

final ArrayBlockingQueue arrayBlockingQueue = new ArrayBlockingQueue(1);

//这里我们增加一个线程去取数据哈,模拟代码没有实际意义啊~
new Thread() {
    @Override
    public void run() {
        super.run();
        try {
            Thread.sleep(1000);
            System.out.println(System.currentTimeMillis() + " 在其它线程中从队列中取数据");
            //取出数据
            arrayBlockingQueue.take();
            Thread.sleep(1000);
            System.out.println(System.currentTimeMillis() + " 在其它线程中从队列中取数据");
           //取出数据
           arrayBlockingQueue.take();
           Thread.sleep(1000);
            System.out.println(System.currentTimeMillis() + " 在其它线程中从队列中取数据");
            //取出数据
            arrayBlockingQueue.take();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}.start();

//下面的code跟上面的栗子是一样
arrayBlockingQueue.put(new Runnable() {
    @Override
    public void run() {
    }
});
System.out.println(System.currentTimeMillis() + " 添加第1个数据");
arrayBlockingQueue.put(new Runnable() {
    @Override
    public void run() {
    }
});
System.out.println(System.currentTimeMillis() + " 添加第2个数据");
arrayBlockingQueue.put(new Runnable() {
    @Override
    public void run() {
    }
});
System.out.println(System.currentTimeMillis() + " 添加第3个数据");

来,关注一下输出结果,可以看到,三个数据都正常的入队和出队啦

1566058033006 添加第1个数据
1566058034009 在其它线程中从队列中取数据
1566058034010 添加第2个数据
1566058035014 在其它线程中从队列中取数据
1566058035014 添加第3个数据
1566058036019 在其它线程中从队列中取数据

阻塞队列在线程池的应用

在 ThreadPoolExecutor 中有一个构造参数 workQueue 就是用到了 BlokingQueue ,那么你有没有好奇 ThreadPoolExecutor 是如何应用这个阻塞机制的呢?

public ThreadPoolExecutor(int corePoolSize,
                          int maximumPoolSize,
                          long keepAliveTime,
                          TimeUnit unit,
                          BlockingQueue<Runnable> workQueue,
                          ThreadFactory threadFactory) {
    this(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue,
         threadFactory, defaultHandler);
}

execute 函数的工作原理

ThreadPoolExecutor 在阻塞队列的应用不外乎就是对需要执行的任务进行缓存,那么就涉及到出队入队两个操作啦,ThreadPoolExecutor 使用主要是用到了以下两个 api

入队

如果当前线程池运行的线程数量小于核心线程数,那么就会将任务添加到队列中,注意哦,这里的入队是用 offer 函数而不是 put 函数,这个是有原因的,首先 offer 函数不会阻塞呀,总不能添加一个任务就把调用者(调用 execute(runnable)的人)给阻塞死吧~这里使用 offer 在队列满时,会直接返回 false 。

workQueue.offer(command)

出队

  • poll(keepAliveTime, TimeUnit.NANOSECONDS)这个方法是针对那些需要在 keepAliveTime 超时后销毁的任务线程(ps:一般是非核心线程啦),这里会一直阻塞等待,直到 keepAliveTime 时间到,如果还没有拿到要执行的任务(ps:说明什么?说明队列没有数据啦~没有人去调用 execute 来提交任务啦~),那么就返回 null,结束使命 game over。
  • workQueue.take()这个方法是针对核心线程的,这个方法会阻塞住哦,直到队列有任务给它(ps:有人通过 execute(runnable)提交任务时,会唤醒它),想到没,这里没有超时机制,所以理所当然,我们的核心线程就不会 game over 啦,它可以一直存活呀,不受 keepAliveTime 影响呀~
Runnable r = timed ?
    workQueue.poll(keepAliveTime, TimeUnit.NANOSECONDS) :
    workQueue.take();

项目地址

https://github.com/liaowjcoder/study4Java/blob/master/03_concurrent/src/main/java/com/example/pool/BlockingQueueDemo.java

本文是笔者学习之后的总结,方便日后查看学习,有任何不对的地方请指正。

记录于 2019年8月18号

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