IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> 自己实现一个阻塞队列 模仿ArrayBlockingQueue -> 正文阅读

[Java知识库]自己实现一个阻塞队列 模仿ArrayBlockingQueue

通过条件队列的等待通知模式来实现 一个基于数组的 有限的阻塞队列

package ThreeYue;

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;

/**
 * 一个阻塞的FIFO队列
 * @param <E>
 */
public class MyBlockingQueue<E> {
    private ReentrantLock lock;
    private Condition notEmpty;//队列空了,消费者线程阻塞到这里
    private Condition notFull;//队列满了,生产者线程阻塞到这里

    private int count;//队列容量
    private int takeIndex;//为了tak使用 用来保证FIFO
    private int putIndex;//为了offer使用
    final Object[] items;

    public MyBlockingQueue(int count) {
        this.count = count;
        this.items = new Object[count];
        lock = new ReentrantLock();
        notEmpty = lock.newCondition();
        notFull = lock.newCondition();
    }

    //加入元素,如果队列满了就阻塞
    public void offer(E item) {
        lock.lock();
        try {
            while(count == items.length) {
                System.out.println("   --生产者被阻塞--");
                notFull.await();
            }
            items[putIndex] = item;
            if (++putIndex == items.length) {
                putIndex = 0;
            }
            count++;
            System.out.println(Thread.currentThread().getName() + " 生产 " + item);
            System.out.println("sc:此时队列中西瓜个数为" + count);
            notEmpty.signal();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            lock.unlock();
        }
    }

    //取出元素,队列空了就阻塞
    public E take() throws InterruptedException {
        lock.lock();
        try {
            while (count == 0) {
                System.out.println("   --消费者被阻塞--");
                notEmpty.await();
            }
            E x = (E) items[takeIndex];
            items[takeIndex] = null;
            if (++takeIndex == items.length) {
                takeIndex = 0;
            }
            count--;
            System.out.println(Thread.currentThread().getName()+" 吃 西瓜");
            System.out.println("xf:此时队列中西瓜个数为" + count);
            notFull.signal();
            return x;
        } finally {
            lock.unlock();
        }
    }

    //测试 一个生产者不断生产 俩个消费者不断消费
    public static void main(String[] args) {
        MyBlockingQueue<String> queue = new MyBlockingQueue<>(3);
        new Thread(()->{
            while(true) {
                queue.offer("西瓜");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"生产者线程1").start();

        new Thread(()->{
            while(true) {
                try {
                    queue.take();
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"消费者线程1").start();

//        new Thread(()->{
//            while(true) {
//                queue.offer("西瓜");
//                try {
//                    Thread.sleep(1000);
//                } catch (InterruptedException e) {
//                    e.printStackTrace();
//                }
//            }
//        },"生产者线程2").start();

        new Thread(()->{
            while(true) {
                try {
                    queue.take();
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"消费者线程2").start();
    }

}

西安今天好热。。。
使用结果

   --生产者被阻塞--
消费者线程1 吃 西瓜
xf:此时队列中西瓜个数为2
生产者线程1 生产 西瓜
sc:此时队列中西瓜个数为3
消费者线程2 吃 西瓜
xf:此时队列中西瓜个数为2
消费者线程1 吃 西瓜
xf:此时队列中西瓜个数为1
生产者线程1 生产 西瓜
sc:此时队列中西瓜个数为2
消费者线程2 吃 西瓜
xf:此时队列中西瓜个数为1
消费者线程1 吃 西瓜
xf:此时队列中西瓜个数为0
   --消费者被阻塞--
生产者线程1 生产 西瓜
sc:此时队列中西瓜个数为1
消费者线程2 吃 西瓜
xf:此时队列中西瓜个数为0
生产者线程1 生产 西瓜
sc:此时队列中西瓜个数为1
消费者线程1 吃 西瓜
xf:此时队列中西瓜个数为0
   --消费者被阻塞--
生产者线程1 生产 西瓜
sc:此时队列中西瓜个数为1
消费者线程1 吃 西瓜
xf:此时队列中西瓜个数为0
   --消费者被阻塞--
生产者线程1 生产 西瓜
sc:此时队列中西瓜个数为1
消费者线程1 吃 西瓜
xf:此时队列中西瓜个数为0
   --消费者被阻塞--
生产者线程1 生产 西瓜
sc:此时队列中西瓜个数为1
消费者线程1 吃 西瓜
xf:此时队列中西瓜个数为0
   --消费者被阻塞--
   --消费者被阻塞--
生产者线程1 生产 西瓜
sc:此时队列中西瓜个数为1
消费者线程2 吃 西瓜
xf:此时队列中西瓜个数为0
  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-03-12 17:18:07  更:2022-03-12 17:20:56 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 8:32:47-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码