返回顶部
首页 > 资讯 > 精选 >JUC并发编程LinkedBlockingQueue队列源码分析
  • 268
分享到

JUC并发编程LinkedBlockingQueue队列源码分析

2023-07-06 00:07:11 268人浏览 独家记忆
摘要

这篇文章主要介绍了JUC并发编程LinkedBlockingQueue队列源码分析的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JUC并发编程LinkedBlockingQueue队列源码分析文章都会有所收获,

这篇文章主要介绍了JUC并发编程LinkedBlockingQueue队列源码分析的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JUC并发编程LinkedBlockingQueue队列源码分析文章都会有所收获,下面我们一起来看看吧。

LinkedBlockingQueue介绍

在JUC包下关于线程安全的队列实现有很多,那么此篇文章讲解LinkedBlockingQueue的实现原理,相信各位读者在线程池中能看到LinkedBlockingQueue或者SynchronousQueue队列来作为储存任务和消费任务的通道。一个并发安全的队列,在多线程中充当着安全的传输任务的责任。

既然是介绍LinkedBlockingQueue,那么从构造方法入手最合适不过。

public LinkedBlockingQueue(int capacity) {    if (capacity <= 0) throw new IllegalArgumentException();    this.capacity = capacity;    // 初始化一个伪节点,让head和last都指向这个伪节点    // 为什么需要伪节点的存在?    // 因为可以保证不会发生极端情况(假设没有伪节点,并且只存在一个节点的情况下,生产者和消费者并发执行就可能出现极端情况)    last = head = new node<E>(null);}

JUC并发编程LinkedBlockingQueue队列源码分析

为什么需要存在伪节点,因为可以保证不会发生极端情况(假设没有伪节点,并且只存在一个节点的情况下,生产者和消费者并发执行就可能出现极端情况,用伪节点就能很好的解决这个极端问题)

transient Node<E> head;private transient Node<E> last;private final ReentrantLock takeLock = new ReentrantLock();private final Condition notEmpty = takeLock.newCondition();private final ReentrantLock putLock = new ReentrantLock();private final Condition notFull = putLock.newCondition();

2套ReentrantLock和对应的condition条件等待队列,很明显目的是为了让生产者和消费者并行,所以就需要一个伪节点处理极端并发情况。

为了,一些没有接触过队列的读者,所以这里还是介绍一下api

API用途注意事项
offer生产者不会阻塞,如果插入失败,或者队列已经满了,直接返回
poll消费者不会阻塞,如果消费失败,或者队列当前为空,直接返回
put生产者会阻塞,如果插入失败或者队列已经满了,阻塞直到插入成功
take消费者会阻塞,如果消费失败或者当前队列为空,阻塞直到消费成功

put方法-生产者

public void put(E e) throws InterruptedException {    // 不能插入null    if (e == null) throw new NullPointerException();    int c = -1;    // 创建插入的节点。    Node<E> node = new Node<E>(e);    // 拿到生产者的对象    final ReentrantLock putLock = this.putLock;    // 拿到全局计数器,注意这里用的是AtomicInteger,所以自增的原子性已经保证。    final AtomicInteger count = this.count;    // 上的是可响应中断锁。    putLock.lockInterruptibly();    try {        // 如果当前队列已经满了,此时我们就要去阻塞,等待队列被消费,我们要被唤醒,醒来生产节点。        while (count.get() == capacity) {            // 进入条件等待队列阻塞。            // 注意,只要阻塞,是会释放锁的,其他生产者线程可以抢到锁。            notFull.await();        }        // 插入到队列尾部        enqueue(node);        // 因为插入了节点,所以全局计数需要+1        // 但是这里请注意细节,getAndIncrement方法返回的是旧值。        c = count.getAndIncrement();        // 这里是一个很sao的点        // 注意,这里只要当前队列没满,唤醒的是生产者的条件等待队列。        // 为什么要这么做?        // 很简单,首先需要考虑,生产者和消费者是并发执行了。         // 其次,只要队列没满就能一直生产,那么队列一旦满了后,后来的线程就都去条件队列阻塞,所以线程生产完一个节点就有必要去唤醒等待的同胞(不管有没有同胞在阻塞,这是义务)        if (c + 1 < capacity)            // 唤醒条件等待队列中头部节点。            notFull.signal();    } finally {        putLock.unlock();    }    // 这里也是一个很sao的点    // 再次强调,getAndIncrement方法是返回的旧值    // 所以当前生产者如果生产的是第一个节点,那么c ==0    // 而队列中没有节点,消费者是要阻塞的    // 也即,这里给队列生产了一个节点,要唤醒消费者去消费节点。    if (c == 0)        signalNotEmpty();}// 插入到队列尾部// 因为ReentrantLock保证了整体的原子性,所以这里细节部分不需要保证原子性了。private void enqueue(Node<E> node) {    // 插入到尾部    last = last.next = node;}

第一次看到这个代码难免会发生震撼,为什么在生产者代码里面唤醒生产者?不是正常写的生产者消费者模型,不都是生产者生产一个唤醒消费者消费吗?怎么这里不一样??????

因为这里生产者和消费者并行处理,当队列满了以后,后来的生产者线程都会去阻塞,所以生产者线程生产完一个节点就有必要去唤醒等待的同胞(不管有没有同胞在阻塞,这是义务)

大致流程如下:

  • 创建Node节点

  • 上生产者锁

  • 如果队列已经满了,就去生产者条件队列阻塞

  • 如果没满,或者唤醒后,就插入到last指针的后面

  • 全局节点计数器+1

  • 如果当前队列还有空间,就唤醒在阻塞的同胞。

  • 释放锁

  • 如果在生产之前队列为空,本次生产后就需要唤醒在阻塞的消费者线程,让他们醒来消费我刚生产的节点

take方法-消费者

public E take() throws InterruptedException {    E x;    int c = -1;    // 全局计数器    final AtomicInteger count = this.count;    // 消费者的锁对象    final ReentrantLock takeLock = this.takeLock;    // 可响应中断锁。    takeLock.lockInterruptibly();    try {        // 如果当前队列中没有节点,此时消费者需要去阻塞,因为不阻塞他只会浪费CPU性能,又消费不到节点。        while (count.get() == 0) {            // 去消费者的条件队列阻塞。            notEmpty.await();        }        // 醒来后,去消费节点。        x = dequeue();        // 给全局计数器-1,但是这里也要注意,返回的是旧值        c = count.getAndDecrement();        // 如果队列中还有节点就唤醒其他消费者去消费节点。        if (c > 1)            notEmpty.signal();    } finally {        takeLock.unlock();    }    // 这里也是一个sao点    // 请注意,这里的c是旧值,因为getAndDecrement返回的是旧值    // 所以,如果当前消费线程消费节点之前队列是满的,当消费完毕后,我有必要去唤醒因为队列满了而阻塞等待的生产者,因为当前已经空出一个空间了。    if (c == capacity)        // 唤醒生产者        signalNotFull();    return x;}// 消费者消费节点// 所以需要HelpGC // 不过这里要注意,head都是指向伪节点。private E dequeue() {    // 拿到头节点,    Node<E> h = head;    // 拿到头节点的next节点,next节点作为下一个head节点。    // 因为head节点是指向伪节点,所以head.next节点就是当前要消费的节点。    Node<E> first = h.next;    // 将当前的头结点的next指向自己。    h.next = h; // help GC    // 设置新的头结点,也即把当前消费的节点做为下次的伪节点    // head节点指向的都是伪节点    head = first;    // 拿到当前消费者想要的数据    E x = first.item;    first.item = null;    return x;}

这里跟put生产者基本思想一致,只不过这里是消费者,因为是生产者消费者并行,所以这里也是唤醒同胞,因为当队列为空所有的消费者都会阻塞,所以每次消费者线程消费完节点后 ,有义务唤醒同胞。

大致流程如下:

  • 拿到全局计数器

  • 上消费者锁

  • 如果当前队列为空,当前消费者线程就要去阻塞

  • 如果不为空,或者被唤醒以后消费节点,把消费的节点作为下一次的伪节点,也即作为head节点

  • 全局计数器-1

  • 唤醒同胞

  • 释放锁

  • 如果在消费之前队列已经满了,那么可能会有生产者线程在阻塞,所以我有义务去唤醒他们

关于“JUC并发编程LinkedBlockingQueue队列源码分析”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“JUC并发编程LinkedBlockingQueue队列源码分析”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注编程网精选频道。

--结束END--

本文标题: JUC并发编程LinkedBlockingQueue队列源码分析

本文链接: https://lsjlt.com/news/356364.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • JUC并发编程LinkedBlockingQueue队列源码分析
    这篇文章主要介绍了JUC并发编程LinkedBlockingQueue队列源码分析的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JUC并发编程LinkedBlockingQueue队列源码分析文章都会有所收获,...
    99+
    2023-07-06
  • JUC并发编程LinkedBlockingQueue队列深入分析源码
    目录LinkedBlockingQueue介绍put方法-生产者take方法-消费者总结LinkedBlockingQueue介绍 在JUC包下关于线程安全的队列实现有很多,那么此篇...
    99+
    2023-05-15
    JUC LinkedBlockingQueue JUC LinkedBlockingQueue队列
  • java并发编程工具类JUC之LinkedBlockingQueue链表队列
    java.util.concurrent.LinkedBlockingQueue 是一个基于单向链表的、范围任意的(其实是有界的)、FIFO阻塞队列。访问与移除操作是在队头进行,添...
    99+
    2024-04-02
  • java并发编程工具类JUC之LinkedBlockingQueue链表队列的示例分析
    小编给大家分享一下java并发编程工具类JUC之LinkedBlockingQueue链表队列的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!java.u...
    99+
    2023-06-15
  • Java并发LinkedBlockingQueue源码分析
    目录简介常量构造方法putawaitisOnSyncQueuesignal简介 LinkedBlockingQueue是一个阻塞的有界队列,底层是通过一个个的Node节点形成的链表...
    99+
    2023-02-12
    Java并发LinkedBlockingQueue Java LinkedBlockingQueue
  • 深入理解Java并发编程之LinkedBlockingQueue队列
    前面一篇文章我们介绍了使用CAS算法实现的非阻塞队列ConcurrentLinedQueue, 下面我们来介绍使用独占锁实现的阻塞队列LinkedBlockingQueue。 Lin...
    99+
    2024-04-02
  • Java并发编程之LinkedBlockingQueue队列怎么使用
    这篇文章主要介绍了Java并发编程之LinkedBlockingQueue队列怎么使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Java并发编程之LinkedBlockingQueue队列怎么使用文章都会有...
    99+
    2023-06-30
  • Java并发编程之JUC并发核心AQS同步队列原理剖析
    目录一、AQS介绍二、AQS中的队列1、同步等待队列2、条件等待队列3、AQS队列节点Node三、同步队列源码分析1、同步队列分析2、同步队列——独占模式源码分析3、同步队列——共享...
    99+
    2024-04-02
  • JUC并发编程中进程与线程的示例分析
    这篇文章将为大家详细讲解有关JUC并发编程中进程与线程的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。进程与线程进程程序由指令和数据组成,但这些指令要运行,数据要读写,就必须将指令加载至 CPU,...
    99+
    2023-06-29
  • Java并发编程之ConcurrentLinkedQueue源码的示例分析
    这篇文章给大家分享的是有关Java并发编程之ConcurrentLinkedQueue源码的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。一、ConcurrentLinkedQueue介绍并编程中,一般需...
    99+
    2023-06-15
  • Java并发编程之CountDownLatch源码解析
    目录一、前言二、使用三、源码分析四、总结一、前言 CountDownLatch维护了一个计数器(还是是state字段),调用countDown方法会将计数器减1,调用await方法会...
    99+
    2024-04-02
  • Java并发编程之LongAdder源码解析
    目录前言源码简介前言 上一篇文章 Java并发编程之原子类(二)中介绍了LongAdder常用的方法,今天我们根据源码来分析一下它的基本实现流程。 This class is usu...
    99+
    2023-05-18
    Java并发LongAdder Java并发
  • Java并发编程之ConcurrentLinkedQueue队列详情
    ConcurrentLinkedQueue JDK中提供了一系列场景的并发安全队列。总的来说,按照实现方式的不同可分为阻塞队列和非阻塞队列,前者使用锁实现,而后则使用CAS非阻塞算法...
    99+
    2024-04-02
  • laravel源码分析队列Queue方法示例
    目录前言队列任务的创建队列任务的分发前言 队列 (Queue) 是 laravel 中比较常用的一个功能,队列的目的是将耗时的任务延时处理,比如发送邮件,从而大幅度缩短 Web 请求...
    99+
    2024-04-02
  • Java多线程并发ReentrantReadWriteLock源码分析
    本篇内容主要讲解“Java多线程并发ReentrantReadWriteLock源码分析”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java多线程并发ReentrantReadWriteLoc...
    99+
    2023-07-02
  • laravel源码分析队列Queue方法怎么用
    本篇内容介绍了“laravel源码分析队列Queue方法怎么用”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!队列任务的创建先通过命令创建一个...
    99+
    2023-06-29
  • Go并发之RWMutex源码分析
    这篇文章主要介绍“Go并发之RWMutex源码分析”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Go并发之RWMutex源码分析”文章能帮助大家解决问题。RWMutex是一个支持并行读串行写的读写锁...
    99+
    2023-07-05
  • Java并发源码分析ConcurrentHashMap线程集合
    目录简介常量构造方法putinitTabletabAtcasTabAthelpTransferputTreeVal锁状态lockRootcontendedLocktreeifyBin...
    99+
    2023-02-01
    Java ConcurrentHashMap Java 线程集合
  • 如何分析Linux消息队列编程
    这期内容当中小编将会给大家带来有关如何分析Linux消息队列编程,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。消息队列,Unix的通信机制之一,可以理解为是一个存放消息(数据)容器。将消息写入消息队列,然...
    99+
    2023-06-28
  • Netty分布式NioEventLoop任务队列执行源码分析
    目录执行任务队列跟进runAllTasks方法:我们跟进fetchFromScheduledTaskQueue()方法回到runAllTasks(long timeoutNanos)...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作