一、基本介绍

ArrayBlcokingQueue,LinkedBlockingQueue是jdk中内置的阻塞队列,网上对它们的分析已经很多,主要有以下几点:

1、底层实现机制不同,ArrayBlcokingQueue是基于数组的,LinkedBlockingQueue是基于链表的;

2、初始化方式不同,ArrayBlcokingQueue是有界的,初始化时必须指定队列的大小;LinkedBlockingQueue可以是无界的,但如果初始化时指定了队列大小,也可以做为有界队列使用;

3、锁机制实现不同,ArrayBlcokingQueue生产和消费使用的是同一把锁,并没有做锁分离;LinkedBlockingQueue中生产、消费分别通过putLock与takeLock保证同步,进行了锁的分离;

使用的过程中,根据应该场景提供了可选插入和删除策略,我们需要掌握和区分

1、插入操作

  1. //队列未满时,返回true;队列满则抛出IllegalStateException(“Queue full”)异常
  2. add(e);
    //队列未满时,直接插入没有返回值;队列满时会阻塞等待,一直等到队列未满时再插入。
  3. put(e);
  4. //队列未满时,返回true;队列满时返回false。非阻塞立即返回。
  5. offer(e);
  6. //设定等待的时间,如果在指定时间内还不能往队列中插入数据则返回false,插入成功返回true。
  7. offer(e, timeout, unit);

2、删除操作

  1. //队列不为空时,返回队首值并移除;队列为空时抛出NoSuchElementException()异常
  2. remove();
  3. //队列不为空返回队首值并移除;当队列为空时会阻塞等待,一直等到队列不为空时再返回队首值。
  4. queue.take();
    //队列不为空时返回队首值并移除;队列为空时返回null。非阻塞立即返回。 
  5. queue.poll();
  6. //设定等待的时间,如果在指定时间内队列还未孔则返回null,不为空则返回队首值 
  7. queue.poll(timeout, unit)

Disruptor框架是由LMAX公司开发的一款高效的无锁内存队列。

Disruptor的最大特点就是高性能,它的内部与众不同的使用了环形队列(RingBuffer)来代替普通的线型队列,相比普通队列环形队列不需要针对性的同步head和tail头尾指针,减少了线程协作的复杂度,再加上它本身基于无锁操作的特性,从而可以达到了非常高的性能;

在使用Disruptor框架时,我们需要注意以下几个方面

1、Disruptor的构造

  1. /**
  2. *
  3. *
  4. * @param eventFactory 定义的事件工厂
  5. * @param ringBufferSize 环形队列RingBuffer的大小,必须是2的N次方
  6. * @param threadFactory 消费者线程工厂
  7. * @param producerType 生产者线程的设置,当你只有一个生产者线程时设置为 ProducerType.SINGLE,多个生产者线程ProducerType.MULTI
  8. * @param waitStrategy 消费者的等待策略
  9. */
  10. public Disruptor(
  11. final EventFactory<T> eventFactory,
  12. final int ringBufferSize,
  13. final ThreadFactory threadFactory,
  14. final ProducerType producerType,
  15. final WaitStrategy waitStrategy)
  16. {
  17. this(
  18. RingBuffer.create(producerType, eventFactory, ringBufferSize, waitStrategy),
  19. new BasicExecutor(threadFactory));
  20. }

上面的消费者等待策略有以下:

BlockingWaitStrategy: 使用锁和条件变量。CPU资源的占用少,延迟大;

SleepingWaitStrategy: 在多次循环尝试不成功后,选择让出CPU,等待下次调度,多次调度后仍不成功,尝试前睡眠一个纳秒级别的时间再尝试。这种策略平衡了延迟和CPU资源占用,但延迟不均匀。

YieldingWaitStrategy: 在多次循环尝试不成功后,通过Thread.yield()让出CPU,等待下次调度。性能和CPU资源占用上较为平衡,但要注意使用该策略时消费者线程最好小于CPU的核心数

BusySpinWaitStrategy: 性能最高的一种,一直不停的自旋等待,获取资源。可以压榨出最高的性能,但会占用最多的CPU资源

PhasedBackoffWaitStrategy: 上面多种策略的综合,CPU资源的占用少,延迟大。

2、handleEventsWith与handleEventsWithWorkerPool的区别

这两个方法区别主要就是在于是否重复消费队列中的消息,前者加载的不同消费者会各自对消息进行消费,各个消费者之间不存在竞争。后者消费者对于队列中的同一条消息不重复消费;

二、性能对比

上面我们对三种阻塞队列做了一个基本的介绍,下面我们分别对它们进行性能上的测试与比对,看下ArrayBlcokingQueue与LinkedBlockingQueue性能上有哪些差别,而Disruptor是否像说的那样具备很高的并发性能

首先我们构造一个加单的消息事件实体

  1. public class InfoEvent implements Serializable {
  2. private static final long serialVersionUID = 1L;
  3. private long id;
  4. private String value;
  5.  
  6. public InfoEvent() {
  7.  
  8. }
  9.  
  10. public InfoEvent(long id, String value) {
  11. this.id = id;
  12. this.value = value;
  13. }
  14.  
  15. public long getId() {
  16. return id;
  17. }
  18.  
  19. public void setId(long id) {
  20. this.id = id;
  21. }
  22.  
  23. public String getValue() {
  24. return value;
  25. }
  26.  
  27. public void setValue(String value) {
  28. this.value = value;
  29. }
  30. }

定义事件工厂

  1. public class InfoEventFactory implements EventFactory<InfoEvent>{
  2. public InfoEvent newInstance() {
  3. return new InfoEvent();
  4. }
  5.  
  6. }

定义Disruptor的消费者

  1. public class InfoEventConsumer implements WorkHandler<InfoEvent> {
  2. private long startTime;
  3. private int cnt;
  4.  
  5. public InfoEventConsumer() {
  6. this.startTime = System.currentTimeMillis();
  7. }
  8.  
  9. @Override
  10. public void onEvent(InfoEvent event) throws Exception {
  11. // TODO Auto-generated method stub
  12. cnt++;
  13.  
  14. if (cnt == DisruptorTest.infoNum) {
  15. long endTime = System.currentTimeMillis();
  16. System.out.println(" 消耗时间: " + (endTime - startTime) + "毫秒");
  17. }
  18.  
  19. }
  20. }

接下来分别针对ArrayBlockingQueue、LinkedBlockingQueue与Disruptor编写测试程序

ArrayBlcokingQueueTest

  1. public class ArrayBlcokingQueueTest {
  2. public static int infoNum = 5000000;
  3. public static void main(String[] args) {
  4. final BlockingQueue<InfoEvent> queue = new ArrayBlockingQueue<InfoEvent>(100);
  5. final long startTime = System.currentTimeMillis();
  6. new Thread(new Runnable() {
  7.  
  8. @Override
  9. public void run() {
  10. int pcnt = 0;
  11. while (pcnt < infoNum) {
  12. InfoEvent kafkaInfoEvent = new InfoEvent(pcnt, pcnt+"info");
  13. try {
  14. queue.put(kafkaInfoEvent);
  15. } catch (InterruptedException e) {
  16. e.printStackTrace();
  17. }
  18. pcnt++;
  19. }
  20. }
  21. }).start();
  22.  
  23. new Thread(new Runnable() {
  24.  
  25. @Override
  26. public void run() {
  27. int cnt = 0;
  28. while (cnt < infoNum) {
  29. try {
  30. queue.take();
  31. } catch (InterruptedException e) {
  32. e.printStackTrace();
  33. }
  34. cnt++;
  35. }
  36. long endTime = System.currentTimeMillis();
  37. System.out.println("消耗时间 : " + (endTime - startTime) + "毫秒");
  38. }
  39. }).start();
  40. }
  41. }

LinkedBlockingQueueTest

  1. public class LinkedBlockingQueueTest {
  2.  
  3. public static int infoNum = 50000000;
  4.  
  5. public static void main(String[] args) {
  6. final BlockingQueue<InfoEvent> queue = new LinkedBlockingQueue<InfoEvent>();
  7. final long startTime = System.currentTimeMillis();
  8. new Thread(new Runnable() {
  9. @Override
  10. public void run() {
  11. int pcnt = 0;
  12. while (pcnt < infoNum) {
  13. InfoEvent kafkaInfoEvent = new InfoEvent(pcnt, pcnt + "info");
  14. try {
  15. queue.put(kafkaInfoEvent);
  16. } catch (InterruptedException e) {
  17. e.printStackTrace();
  18. }
  19. pcnt++;
  20. }
  21. }
  22. }).start();
  23.  
  24. new Thread(new Runnable() {
  25.  
  26. @Override
  27. public void run() {
  28. int cnt = 0;
  29. while (cnt < infoNum) {
  30. try {
  31. queue.take();
  32. } catch (InterruptedException e) {
  33. e.printStackTrace();
  34. }
  35. cnt++;
  36. }
  37. long endTime = System.currentTimeMillis();
  38. System.out.println("消耗时间: " + (endTime - startTime) + "毫秒");
  39. }
  40. }).start();
  41. }
  42.  
  43. }

DisruptorTest

  1. public class DisruptorTest {
  2. public static int infoNum = 5000000;
  3. @SuppressWarnings("unchecked")
  4. public static void main(String[] args) {
  5. InfoEventFactory factory = new InfoEventFactory();
  6. int ringBufferSize = 65536; //数据缓冲区的大小 必须为2的次幂
  7.  
  8. /**
  9. *
  10. * factory,定义的事件工厂
  11. * ringBufferSize,环形队列RingBuffer的大小,必须是2的N次方
  12. * ProducerType,生产者线程的设置,当你只有一个生产者线程时设置为 ProducerType.SINGLE,多个生产者线程ProducerType.MULTI
  13. * waitStrategy,消费者的等待策略
  14. *
  15. */
  16. final Disruptor<InfoEvent> disruptor = new Disruptor<InfoEvent>(factory, ringBufferSize,
  17. DaemonThreadFactory.INSTANCE, ProducerType.SINGLE, new YieldingWaitStrategy());
  18.  
  19. InfoEventConsumer consumer = new InfoEventConsumer();
  20. disruptor.handleEventsWithWorkerPool(consumer);
  21. disruptor.start();
  22. new Thread(new Runnable() {
  23. @Override
  24. public void run() {
  25. RingBuffer<InfoEvent> ringBuffer = disruptor.getRingBuffer();
  26. for (int i = 0; i < infoNum; i++) {
  27. long seq = ringBuffer.next();
  28. InfoEvent infoEvent = ringBuffer.get(seq);
  29. infoEvent.setId(i);
  30. infoEvent.setValue("info" + i);
  31. ringBuffer.publish(seq);
  32. }
  33. }
  34. }).start();
  35. }
  36. }

我们在十万、百万、千万三个数量级上,分别对ArrayBlockingQueue,LinkedBlockingQueue初始化为无界和有界队列,Disruptor的BlockingWaitStrategy和YieldingWaitStrategy,进行三次测试,生产者与消费者均在单线程模式下运行,对结果进行统计记录;

测试环境:

操作系统:win7 64位,CPU:Intel Core i7-3250M 2.9GHz ,内存:8G,JDK:1.8,disruptor版本:3.4.2

五十万数据

 

第一次

第二次

第三次

ArrayBlcokingQueue

229ms

233ms

253ms

LinkedBlockingQueue(无界)

211ms

207ms

202ms

LinkedBlockingQueue(有界)

265ms

207ms

256ms

Disruptor(BlockingWaitStrategy)

71ms

56ms

65ms

Disruptor(YieldingWaitStrategy)

56ms

48ms

49ms

五百万数据

 

第一次

第二次

第三次

ArrayBlcokingQueue

1530ms

1603ms

1576ms

LinkedBlockingQueue(无界)

1369ms

1390ms

1409ms

LinkedBlockingQueue(有界)

1408ms

1397ms

1494ms

Disruptor(BlockingWaitStrategy)

345ms

363ms

357ms

Disruptor(YieldingWaitStrategy)

104ms

108ms

107ms

五千万数据

第一次

第二次

第三次

ArrayBlcokingQueue

14799ms

14928ms

15122ms

LinkedBlockingQueue(无界)

14226ms

14008ms

13518ms

LinkedBlockingQueue(有界)

14039ms

14434ms

13839ms

Disruptor(BlockingWaitStrategy)

2972ms

2910ms

2848ms

Disruptor(YieldingWaitStrategy)

699ms

742ms

698ms

  1. 然后我对程序进行了修改,让测试程序持续运行,每五千万输出一次,对运行期间CPU和内存使用情况进行了记录
  2.  
  3. ArrayBlcokingQueue
  4.  
  5. LinkedBlockingQueue(无界)
  6.  
  7. LinkedBlockingQueue(有界)
  8.  
  9. DisruptorBlockingWaitStrategy
  10.  

  11. DisruptorYieldingWaitStrategy
  12.  
  13. 从上面的测试中我们可以看到ArrayBlcokingQueueLinkedBlockingQueue性能上区别不是很大,LinkedBlockingQueue由于读写锁的分离,平均性能会稍微好些,但差距并不明显。
    Disruptor性能表现突出,特别是随着数据量的增大,优势会越发明显。同时在单线程生产和消费的应用场景下,相比jdk内置的阻塞队列,CPUGC的压力反而更小。
  14.  
  15. 三、总结

1、ArrayBlcokingQueue与LinkedBlockingQueue,一般认为前者基于数组实现,初始化后不需要再创建新的对象,但没有进行锁分离,所以内存GC压力较小,但性能会相对较低;后者基于链表实现,每次都需要创建  一个node对象,会存在频繁的创建销毁操作,GC压力较大,但插入和删除数据是不同的锁,进行了锁分离,性能会相对较好;从测试结果上看,其实两者性能和GC上差别都不大,在实际运用过程中,我认为一般场景下ArrayBlcokingQueue的性能已经足够应对,处于对GC压力的考虑,及潜在的OOM的风险我建议普通情况下使用ArrayBlcokingQueue即可。当然你也可以使用LinkedBlockingQueue,从测试结果上看,它相比ArrayBlcokingQueue性能上有有所提升但并不明显,结合gc的压力和潜在OOM的风险,所以结合应用的场景需要综合考虑。

  1. 2Disruptor做为一款高性能队列框架,确实足够优秀,在测试中我们可以看到无论是性能和GC压力都远远好过ArrayBlcokingQueueLinkedBlockingQueue;如果你追求更高的性能,那么Disruptor是一个很好的选择。
    但需要注意的是,你需要结合自己的硬件配置和业务场景,正确配置Disruptor,选择合适的消费策略,这样不仅可以获取较高的性能,同时可以保证硬件资源的合理分配。
  1. 3、对这三种阻塞队列的测试,并不是为了比较孰优孰劣,主要是为了加强理解,实际的业务应用需要根据情况合理进行选择。这里只是结合自己的使用,对它们进行一个简单的总结,并没有进行较深入的探究,如有错误的的地方还请指正与海涵。

关注微信公众号,查看更多技术文章。

  1.  

ArrayBlcokingQueue,LinkedBlockingQueue与Disruptor三种队列对比与分析的更多相关文章

  1. Android自动化测试中AccessibilityService获取控件信息(2)-三种方式对比

    Android自动化测试中AccessibilityService获取控件信息(2)-三种方式对比   上一篇文章: Android自动化测试中AccessibilityService获取控件信息(1 ...

  2. c#之冒泡排序的三种实现和性能分析

    冒泡排序算法是我们经常见到的尤其是子一些笔试题中. 下面和大家讨论c#中的冒泡排序,笔者提供了三种解决方案,并且会分析各自的性能优劣. 第一种估计大家都掌握的,使用数据交换来实现,这种就不多说了,园子 ...

  3. 三种工厂模式的分析以及C++实现

    三种工厂模式的分析以及C++实现 以下是我自己学习设计模式的思考总结. 简单工厂模式 简单工厂模式是工厂模式中最简单的一种,他可以用比较简单的方式隐藏创建对象的细节,一般只需要告诉工厂类所需要的类型, ...

  4. Mysql Binlog三种格式介绍及分析【转】

    一.Mysql Binlog格式介绍       Mysql binlog日志有三种格式,分别为Statement,MiXED,以及ROW! 1.Statement:每一条会修改数据的sql都会记录在 ...

  5. C2B电商三种主要模式的分析_数据分析师

    C2B电商三种主要模式的分析_数据分析师 在过去的一年中电商领域血雨腥风,尤其是天猫.京东.苏宁.当当.易讯等B2C电商打得不亦乐乎.而随着B2C领域竞争进入白热化阶段,C2B模式也在天猫" ...

  6. Day008 三种初始化及内存分析

    三种初始化和内存分析 Java内存分析: 堆: 存放new的对象和数组. 可以被所有的线程共享,不会存放别的对象引用. 栈: 存放基本变量类型(会包含这个基本类型的具体数值). 引用对象的变量(会存放 ...

  7. Apache2 三种MPM对比分析

    就最新版本的Web服务器Apache(版本是Apache 2.4.10,发布于2014年7月21日)来说,一共有三种稳定的MPM(Multi-Processing Module,多进程处理模块)模式. ...

  8. APP开发的三种技术对比

    目前来说主流的App开发方式有三种:Native App .Web App.Hybird App.下面我们来分析一下这三种App开发方式的优劣对比: 一 :Native App 即 原生App开发 优 ...

  9. Python队列的三种队列方法

    今天讲一下队列,用到一个python自带的库,queue 队列的三种方法有: 1.FIFO先入先出队列(Queue) 2.LIFO后入先出队列(LifoQueue) 3.优先级队列(PriorityQ ...

随机推荐

  1. Oracle自我补充之trunc()函数使用介绍

    oracle trunc函数使用介绍 核心提示:oracle trunc函数使用介绍 1.TRUNC(for dates) TRUNC函数为指定元素而截去的日期值. 其具体的语法格式如下: TRUNC ...

  2. JAVA中关于对像的读写

    /** * 针对对象的文件读写 */ //导入包 import java.io.File; import java.io.FileInputStream; import java.io.FileNot ...

  3. 2.5 SeleniumBuilder辅助定位元素

    前言对于用火狐浏览器的小伙伴们,你还在为定位元素而烦恼嘛?上古神器Selenium Builder来啦,哪里不会点哪里,妈妈再也不用担心我的定位元素问题啦!(但是也不是万能,基本上都能覆盖到) 2.5 ...

  4. JavaScript中的内置对象-8--2.String-符串对象 方法; 截取方法; 综合应用; 其他方法;

    JavaScript内置对象-2String(字符串) 学习目标 1.掌握字符串对象 方法: charAt() charCodeAt() indexOf() lastIndextOf() charAt ...

  5. 安装12C小问题及pdb表空间配置

    安装12C小问题及pdb表空间配置 一.安装 1.RPM包 #安装12C需要安装的rpm包,官网搜索,做个记录 bc binutils-2.23.52.0.1-12.el7(x86_64) compa ...

  6. POJ 3468:A Simple Problem with Integers(线段树区间更新模板)

    A Simple Problem with Integers Time Limit: 5000MS   Memory Limit: 131072K Total Submissions: 141093 ...

  7. StackExchange.Redis和Log4Net构建日志

    利用StackExchange.Redis和Log4Net构建日志队列   简介:本文是一个简单的demo用于展示利用StackExchange.Redis和Log4Net构建日志队列,为高并发日志处 ...

  8. 小米4c刷LineageOS

    注意,本文仅限于小米4c,其他手机仅可参考步骤.如下rom,su,gapps包的下载都是小米4c的,深刷miflash也仅适用于小米手机.准备工作:请自行备份好手机内的个人资料. 电脑环境,usb驱动 ...

  9. YIT-CTF—社工类

    下载图片

  10. mysql插入操作跳过(ignore)、覆盖(replace into)、更新(on duplicate key)

    原帖地址:http:.html .insert ignore into 当插入数据时,如出现错误时,如重复数据,将不返回错误,只以警告形式返回.所以使用ignore请确保语句本身没有问题,否则也会被忽 ...