阻塞队列-双锁实现

 我们原来单锁的话 容易  比较双锁来讲效率较低,而且size变量是线程不安全的,我们用双锁来进行优化

队列接口

public interface BlockingQueue <E>{
    void offer(E e) throws InterruptedException;
    Boolean offer(E e,long timeout) throws InterruptedException;
    E poll() throws InterruptedException;
}

阻塞队列双锁实现类 

我们的双锁实现的精髓 就是

我们添加元素之后,获得poll的锁之后,只唤醒一次poll线程,如果还有其他poll线程,由poll线程自己判断是否需要唤醒其他的poll线程

相应的  我们删除元素之后,获得offer锁之后,只唤醒一次offer线程,由offer线程内部去判断是否需要唤醒其他offer线程

public class ArrayBlockQueueTwoLock<E> implements BlockingQueue<E> {
    private final E[] array;
    private int head;//记录出队时候的头指针
    private int tail;//记录入队的指针
    //记录数组 个数  但是多线程下 单纯的int 是线程不安全的  我们换成线程安全的
    AtomicInteger size = new AtomicInteger(0);
    //双锁 实现 我们让  offer 线程 和poll线程分别持有一把锁
    private ReentrantLock tailLock = new ReentrantLock();//offer 线程的锁

    private ReentrantLock headLock = new ReentrantLock();//poll线程的锁
    private Condition headWait = headLock.newCondition();//控制 入队的 同步队列
    private Condition tailWait = tailLock.newCondition();//控制出队的同步队列

    public ArrayBlockQueueTwoLock(int capacity) {
        array = (E[]) new Object[capacity];
    }

    private Boolean isFull() {
        return size.get() == array.length;
    }

    private Boolean isEmpty() {
        return size.get() == 0;
    }

    @Override
    public String toString() {
        return "ArrayBlockingQueue{" +
                "array=" + Arrays.toString(array) +
                '}';
    }

    @Override
    public void offer(E e) throws InterruptedException {
        int count;//我们单纯的双锁 进行 判断 优化的还是不够好
        // 我们加一个计数器  加入同时有多个线程阻塞
        // 我们通过计数器判断 是应该单个线程 直接唤醒多个剩余的offer线程 还是单个线程只加锁一次
        tailLock.lockInterruptibly();
        try {
            while (isFull()) {
                tailWait.await();//如果满了 加入等待队列
            }
            array[tail] = e;
            if (++tail == array.length) {
                tail = 0;
            }
            count = size.incrementAndGet();//返回原来的数 并且 返回之后加1
            //如果有三个 offer 线程等待加入  发现 size为空 那我们 直接唤醒剩下的offer线程
            if (count < array.length) {
                tailWait.signal();
            }
        } finally {
            tailLock.unlock();
        }
        //如果 不为空队列的时候 加入一个 唤醒  删除的线程
        //我们唤醒 删除线程只需要唤醒 一次 即可
        // 后续的 让 删除的线程 自己去判断是否应该唤醒 其他的删除线程
        if(count==1){
            headLock.lockInterruptibly();
            try {
                headWait.signal();
            } finally {
                headLock.unlock();
            }
        }

    }

    @Override
    public E poll() throws InterruptedException {
        E e;
        int count;
        headLock.lockInterruptibly();
        try {
            while (isEmpty()) {
                headWait.await();//如果为空 一直等待
            }
            e = array[head];
            array[head] = null;
            if (++head == array.length) {
                head = 0;
            }
            count = size.decrementAndGet();
            //如果这时候排了几个删除的线程 我们发现 count 不为空 那么直接唤醒删除的线程
            if (count > 0) {
                headWait.signal();
            }
        } finally {
            headLock.unlock();
        }
        //如果队列满的时候 删除一个 元素之后  需要唤醒等待的 offer线程  我们也只需要唤醒一次就行
        // 写到 poll锁中容易死锁 所以我们把锁分开
        if(count==array.length-1){
            tailLock.lockInterruptibly();
            try {
                tailWait.signal();
            } finally {
                tailLock.unlock();
            }
        }
        return e;
    }

    @Override
    public Boolean offer(E e, long timeout) throws InterruptedException {
        return null;
    }

}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

三氧化真

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值