阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

89 篇文章 0 订阅

 

本文聊一下 JUC 下的 LinkedBlockingQueue 队列,先说说 LinkedBlockingQueue 队列的特点,然后再从源码的角度聊一聊 LinkedBlockingQueue 的主要实现~

LinkedBlockingQueue 有以下特点:

  • 「LinkedBlockingQueue 是阻塞队列,底层是单链表实现的」~
  • 「元素从队列尾进队,从队列头出队,符合FIFO」~
  • 「可以使用 Collection 和 Iterator 两个接口的所有操作,因为实现了两者的接口」~
  • 「LinkedBlockingQueue 队列读写操作都加了锁,但是读写用的是两把不同的锁,所以可以同时读写操作」~

LinkedBlockingQueue 队列继承了 AbstractQueue 类,实现了 BlockingQueue 接口,LinkedBlockingQueue 主要有以下接口:

//将指定的元素插入到此队列的尾部(如果立即可行且不会超过该队列的容量)
//在成功时返回 true,如果此队列已满,则抛IllegalStateException。 
boolean add(E e); 

//将指定的元素插入到此队列的尾部(如果立即可行且不会超过该队列的容量) 
// 将指定的元素插入此队列的尾部,如果该队列已满, 
//则在到达指定的等待时间之前等待可用的空间,该方法可中断 
boolean offer(E e, long timeout, TimeUnit unit) throws InterruptedException; 

//将指定的元素插入此队列的尾部,如果该队列已满,则一直等到(阻塞)。 
void put(E e) throws InterruptedException; 

//获取并移除此队列的头部,如果没有元素则等待(阻塞), 
//直到有元素将唤醒等待线程执行该操作 
E take() throws InterruptedException; 

//获取并移除此队列的头,如果此队列为空,则返回 null。 
E poll();
//获取并移除此队列的头部,在指定的等待时间前一直等到获取元素, //超过时间方法将结束
E poll(long timeout, TimeUnit unit) throws InterruptedException; 

//从此队列中移除指定元素的单个实例(如果存在)。 
boolean remove(Object o); 

//获取但不移除此队列的头元素,没有则跑异常NoSuchElementException 
E element(); 

//获取但不移除此队列的头;如果此队列为空,则返回 null。 
E peek(); 

LinkedBlockingQueue 队列的读写方法非常的多,但是常用的是 put()、take()方法,因为它们两是阻塞的,所以我们就从源码的角度来聊一聊 LinkedBlockingQueue 队列中这两个方法的实现。

先来看看 put()方法,源码如下:

public void put(E e) throws InterruptedException {
    if (e == null) throw new NullPointerException();
    // 预先设置 c 的值为 -1,表示失败
    int c = -1;
    Node<E> node = new Node<E>(e);
    // 获取写锁
    final ReentrantLock putLock = this.putLock;
    // 获取当前队列的大小
    final AtomicInteger count = this.count;
    // 设置可中断锁
    putLock.lockInterruptibly();
    try {
        // 队列满了
        // 当前线程阻塞,等待其他线程的唤醒(其他线程 take 成功后就会唤醒此处线程)
        while (count.get() == capacity) {
            // 无限期等待
            notFull.await();
        }
        // 新增到队列尾部
        enqueue(node);
        // 获取当前的队列数
        c = count.getAndIncrement();
        // 如果队列未满,尝试唤醒一个put的等待线程
        if (c + 1 < capacity)
            notFull.signal();
    } finally {
        // 释放锁
        putLock.unlock();
    }
    if (c == 0)
        signalNotEmpty();
}

put()方法的源码并不难,非常容易就看懂,put()方法的过程大概如下:

  • 1、先加锁,保证容器的并发安全~
  • 2、队列新增数据,将数据追加到队列尾部~
  • 3、新增时,如果队列满了,当前线程是会被阻塞的,等待被唤醒~
  • 4、新增数据成功后,在适当时机,会唤起 put 的等待线程(队列不满时),或者 take 的等待线程(队列不为空时),这样保证队列一旦满足 put 或者 take 条件时,立马就能唤起阻塞线程,继续运行,保证了唤起的时机不被浪费offer 就有两两种,一种是直接返回 false,另一种是超过一定时间后返回 false~
  • 5、释放锁~

其他的新增方法,例如 offer,可以查看源码,跟put() 方法大同小异,相差不大~

再来看看 take()方法,源码如下:

public E take() throws InterruptedException {
    E x;
    // 默认负数
    int c = -1;
    // 当前链表的个数
    final AtomicInteger count = this.count;
    //获取读锁
    final ReentrantLock takeLock = this.takeLock;
    takeLock.lockInterruptibly();
    try {
        // 当队列为空时,阻塞,等待其他线程唤醒 
        while (count.get() == 0) {
            notEmpty.await();
        }
        // 从队列的头部拿出一个元素
        x = dequeue();
        //减一操作,C比真实的队列数据大一
        c = count.getAndDecrement();
        // c 大于 0 ,表示队列有值,可以唤醒之前被阻塞的读线程
        if (c > 1)
            notEmpty.signal();
    } finally {
        // 释放锁
        takeLock.unlock();
    }
    // 队列未满,可以唤醒 put 等待线程~
    if (c == capacity)
        signalNotFull();
    return x;
}

take()方法跟 put() 方法类似,是一个相反的操作,我就不做过多的说明了~

以上就是 LinkedBlockingQueue 队列的简单源码解析,希望对你的面试或者工作有所帮助,感谢你的阅读~

最后,再附上我历时三个月总结的 Java 面试 + Java 后端技术学习指南,这是本人这几年及春招的总结,目前,已经拿到了大厂offer,拿去不谢!

如何拿下阿里等大厂的offer的呢,今天分享一个秘密武器,资深架构师整理的350页java开发面试真题和开发必备面试题详解(技术+人事),面试时面试官必问的知识点,篇章包括了很多知识点,其中包括了有基础知识、Java集合、JVM、多线程并发、spring原理、微服务、Netty 与RPC 、Kafka、日记、设计模式、Java算法、数据库、Zookeeper、分布式缓存、数据结构等等。

由于整个文档比较全面,内容比较多,篇幅的限制,文章中分享没有全部附上详细的解析,但是整理成了一份详细的PDF文档可分享给大家,文末有免费领取方式。

JAVA面试题手册

一性能优化面试专栏

 

二微服务架构面试专栏

 

 

三并发编程高级面试专栏

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

四、开源框架面试题专栏

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

五、分布式面试专栏

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

java开发必备面试题详解(技术+人事)

数据结构和算法

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

jvm

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

 

 

Java核心基础

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

 

 

多线程/高并发

 

 

 

集合框架

 

 

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

IO流

 

 

数据库

 

 

javaWEB

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

企业级框架

 

 

Linux

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

软件工程与设计模式

 

人事问题

阿里P9整理Java 高频面试题聊一聊 JUC 下的 LinkedBlockingQueue

 

说明,为了节省大家的时间和提高学习效率,-些过时知识点和被笔试概率极低的题目不再被收录和分析。

回答问题的思路:先正面叙述一些基本的核心知识,然后描述一些特殊的东西,最后再来一些锦上添花的东西。要注意有些不是锦上添花,而是画蛇添足的东西,不要随便写上。把答题像写书一样写。我要回答一个新技术的问题大概思路和步骤是:我们想干什么,怎么干,干的过程中遇到了什么问题,现在用什么方式来解决。其实我们讲课也是这样一个思路。

需要获取这两份PDF资料的小伙伴可以转发+关注后私信(学习)即可免费获取

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值