前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >java进阶|LinkedBlockingQueue源码分析

java进阶|LinkedBlockingQueue源码分析

作者头像
码农王同学
发布2020-06-04 11:11:29
3750
发布2020-06-04 11:11:29
举报
文章被收录于专栏:后端Coder后端Coder

现在是2020/05/18:23:12分,是的,马上就要到凌晨了,然而我才开始写这篇文章,为什么这么晚写这篇文章,不困吗,或许是,或许不是,其实在我刷完抖音之后脑海里想的就是分析一下LinkedBlockingQueue的源码吧,至于为什么要这么晚还去分析,有这个必要吗,或许是自己喜欢这个点有点久了。

一般你们遇到的每一篇文章都是经过我最少一个周之前想写的内容,所以文章出现的时候,我自己在心里也沉淀了很久才发出来,这样就会比较对自己友好一点。

是的,是人都会有私心,我喜欢分享,但我不会将很私密的事情去分享,因为互联网上什么人都有,没有必要将自己喜欢的东西分享出来,文章只会分享一些技术相关的内容和自己思考的一些事情,所以你看到的内容也就是我喜欢分享的内容,好了,闲扯就不说了,接下来我要开始分析LinkedBlockingQueue了。

按照自己的风格,接下来先看下LinkedBlockingQueue的继承接口和实现接口吧。

代码语言:javascript
复制
public class LinkedBlockingQueue<E> extends AbstractQueue<E>
        implements BlockingQueue<E>, java.io.Serializable {}

java中的继承,封装,多态都在源码里面体现的比较友好,所以这里就不过多介绍了,想了解这些内容的可以去搜索了解一下,一般我们创建一个容器都是从new关键字开始的,容器的大小一般支持自定义的方式。

代码语言:javascript
复制
public LinkedBlockingQueue() {
        this(Integer.MAX_VALUE);
    }

当我们手动new一个容器时,初始大小分配好了,这就是整形数值的最大值,即默认开辟了这么大的数组空间,即内存空间,是不是有点浪费呢?自己思考思考。

代码语言:javascript
复制
 public LinkedBlockingQueue(int capacity) {
        if (capacity <= 0) throw new IllegalArgumentException();
        this.capacity = capacity;
        last = head = new Node<E>(null);
    }

其实,我们也可以自己手动容器的大小,,但是不能小于等于0,这里就进行了前置校验,抛出非法异常,然后将初始值赋值给数组容量,因为数据是存放在节点当中的,所以这里就暂时将new Node()的值设置为了null。

我们使用容器要干什么?装填数据呗,就是常用的offer(),put()方法,这里我们先介绍一下put()方法的源码吧。

代码语言:javascript
复制
 public void put(E e) throws InterruptedException {
        if (e == null) throw new NullPointerException();
        //若装填的数据为null,直接抛出空指针异常
        int c = -1;
        Node<E> node = new Node<E>(e);
        final ReentrantLock putLock = this.putLock;//获取putLock锁
        final AtomicInteger count = this.count;//获取当前队列的元素个数
        putLock.lockInterruptibly();//这个锁是可中断的
        try {
        //这句话就是表明put方法是一个阻塞性的方法
            while (count.get() == capacity) {
                notFull.await();
            }
            //如队列操作
            enqueue(node);
            c = count.getAndIncrement();
            if (c + 1 < capacity)
                notFull.signal();
        } finally {
        //最后在finally语句块进行释放锁
            putLock.unlock();
        }
        if (c == 0)
            signalNotEmpty();
    }

首先这是一个线程安全的方法,为什么这么说呢?看到lock了没,加锁了,同一时刻只能有一个线程进行操作,保证了多线程下共享数据的安全。

LinkedBlockingQueue不允许队列的元素为null,所以,这里在入队列之前就进行了前置校验,若元素e为空,则直接抛出NPE异常,阻断程序的运行,在入队列之前先判断队列是否已满,若已满则等待,这也是为什么之前有的面试官总是会问到put()方法和其它添加元素方法的区别,是不是看过源码之后一目了然。

代码语言:javascript
复制
private void enqueue(Node<E> node) {
      
        last = last.next = node;
    }

入队列很简单,就是将新增节点数据直接连接到队列的尾部,这样就保证了队列的先入先出特点。

好了,put()方法的过程到这里就结束了,接下来我们在看下其它方法吧。

就暂时先看下take()方法的实现过程吧,首先这个take()方法也是一个阻塞性方法,队列若没有数据,则直接等待。

代码语言:javascript
复制
 public E take() throws InterruptedException {
        E x;
        int c = -1;
        final AtomicInteger count = this.count;//获取当前队列元素的个数
        final ReentrantLock takeLock = this.takeLock;//获取takeLock锁
        takeLock.lockInterruptibly();//进行加锁操作
        try {
            while (count.get() == 0) {//判断队列的元素个数是否为0,若为0则等待
                notEmpty.await();
            }
            x = dequeue();//出队列操作
            c = count.getAndDecrement();
            if (c > 1)
                notEmpty.signal();
        } finally {
        //释放锁操作
            takeLock.unlock();
        }
        if (c == capacity)
            signalNotFull();
        return x;
    }

首先判断当前队列的元素个数是否为0,若为0,则等待,这里就基于condition的await()方法实现的等待机制。然后就是dequeue()方法了,

代码语言:javascript
复制
 private E dequeue() {
        Node<E> h = head;
        Node<E> first = h.next;
        h.next = h; // help GC
        head = first;
        E x = first.item;
        first.item = null;
        return x;
    }

出队列就将数据置为null,便于GC进行不可用数据的回收,其实在你看这部分时了解一下jvm是很必要的,一般队列和栈都很详细,就是我想获取队列的队首元素,但是我又不想让它出队列,这样就提供了peek()方法。

代码语言:javascript
复制
public E peek() {
        if (count.get() == 0)
            return null;
        final ReentrantLock takeLock = this.takeLock;
        takeLock.lock();
        try {
            Node<E> first = head.next;
            if (first == null)
                return null;
            else
                return first.item;
        } finally {
            takeLock.unlock();
        }
    }

这里,首先判断队列的元素个数是否为0,若为0,则直接返回null,否则,加锁,然后获取 队列队首元素,这样就获取了你想要的队首元素,关于这部分,其实你理解了什么是锁,什么是原子类,这个分析过程和分析ArrayList源码基本上一样,没有看过我的源码分析的请历史文章进行查找。

其实熟悉我的文章的读者都知道我分析集合容器时总是喜欢分析它的isEmpty()方法和clear()方法,因为我觉得这个对于我理解集合太重要了。

代码语言:javascript
复制
 public void clear() {
        fullyLock();
        try {
            for (Node<E> p, h = head; (p = h.next) != null; h = p) {
                h.next = h;
                p.item = null;
            }
            head = last;
            if (count.getAndSet(0) == capacity)
                notFull.signal();
        } finally {
            fullyUnlock();
        }
    }

循环迭代队列的元素,然后将其置为null,等待GC进行数据的回收,这样队列就清空了,然后将表示队列元素个数清零就可以了,整个过程的加锁和释放就不过多介绍了,到这里差不多队列的内容就介绍完了,方法太多,我这里就简单介绍了一部分方法,其它方法大差不差没有什么区别,所里这里就不过多介绍了,

代码语言:javascript
复制
public int size() {
        return count.get();
    }

这个count是原子的,即使是多线程操作下也能保证原子性。差点忘了分析这个方法了,判断元素是否在队列里。

代码语言:javascript
复制
 public boolean contains(Object o) {
        if (o == null) return false;
        fullyLock();
        try {
            for (Node<E> p = head.next; p != null; p = p.next)
                if (o.equals(p.item))
                    return true;
            return false;
        } finally {
            fullyUnlock();
        }
    }

循环迭代队列里的元素值,与其进行比较,若相等则返回true,否则返回false,时间复杂度为O(n),不了解时间复杂度,这里不再继续说了,需要的可以自行了解。

最后,在看下如何转为数组的方法吧,这个方法比较特殊,因为它没有调用底层的拷贝,利用的确实新开辟了一个数组,然后将数组进行填充到新的数组里面。

代码语言:javascript
复制
 public Object[] toArray() {
        fullyLock();
        try {
            int size = count.get();
            Object[] a = new Object[size];
            int k = 0;
            for (Node<E> p = head.next; p != null; p = p.next)
                a[k++] = p.item;
            return a;
        } finally {
            fullyUnlock();
        }
    }

整个过程也是加锁和释放锁的过程,所以这就是一个线程安全的容器类,到这里就结束了,时间也不早了,0:02了,好了,喜欢的内容到这里点个在看呗,到这里就结束了,后面想要分享内容再继续分享。

我喜欢分享,你喜欢阅读@WwpwW

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2020-06-01,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 码农王同学 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
容器服务
腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档