Java EE 初阶---多线程(三)

news2025/1/10 20:27:47

 五、阻塞队列

目录

 五、阻塞队列

5.1 阻塞队列是什么 ?

5.1.1 生产者消费者模型

​编辑

 5.1.2 标准库中的阻塞队列

 5.1.3 消息队列

 5.1.4 消息队列的作用

 5.2 实现一个阻塞队列

虚假唤醒

六、线程池

6.1 线程池是什么?

 6.2 怎么使用线程池?

6.2.1 JDK给我们提供了一些方法来创建线程池(不建议使用)

6.2.2 工厂模式

6.2.3 自定义一个线程池

为什么不推荐使用系统自带的线程池?



5.1 阻塞队列是什么 ?

阻塞队列是一种特殊的队列 . 也遵守 " 先进先出 " 的原则 .
阻塞队列能是一种线程安全的数据结构 , 并且具有以下特性 :
  • 当队列满的时候, 继续入队列就会阻塞, 直到有其他线程从队列中取走元素.
  • 当队列空的时候, 继续出队列也会阻塞, 直到有其他线程往队列中插入元素.
  • 阻塞队列的一个典型应用场景就是 "生产者消费者模型". 这是一种非常典型的开发模型.

举个栗子:

5.1.1 生产者消费者模型

生产者消费者模式就是通过一个容器来解决生产者和消费者的强耦合问题。
生产者和消费者彼此之间不直接通讯,而通过阻塞队列来进行通讯,所以生产者生产完数据之后不用等待消费者处理,直接扔给阻塞队列,消费者不找生产者要数据,而是直接从阻塞队列里取
1) 阻塞队列就相当于一个缓冲区,平衡了生产者和消费者的处理能力 .
比如在 " 秒杀 " 场景下 , 服务器同一时刻可能会收到大量的支付请求 . 如果直接处理这些支付请求 , 服务器可能扛不住( 每个支付请求的处理都需要比较复杂的流程 ). 这个时候就可以把这些请求都放到一个阻塞队列中, 然后再由消费者线程慢慢的来处理每个支付请求 .
这样做可以有效进行 " 削峰 ", 防止服务器被突然到来的一波请求直接冲垮 .
2) 阻塞队列也能使生产者和消费者之间 解耦 .
比如过年一家人一起包饺子 . 一般都是有明确分工 , 比如一个人负责擀饺子皮 , 其他人负责包 . 擀饺子皮的人就是 " 生产者 ", 包饺子的人就是 " 消费者 ".
擀饺子皮的人不关心包饺子的人是谁 ( 能包就行 , 无论是手工包 , 借助工具 , 还是机器包 ), 包饺子的人也不关心擀饺子皮的人是谁( 有饺子皮就行 , 无论是用擀面杖擀的 , 还是拿罐头瓶擀 , 还是直接从超市买的).

 5.1.2 标准库中的阻塞队列

Java 标准库中内置了阻塞队列 . 如果我们需要在一些程序中使用阻塞队列 , 直接使用标准库中的即可 .
  • BlockingQueue 是一个接口. 真正实现的类是 LinkedBlockingQueue.
  • put 方法用于阻塞式的入队列, take 用于阻塞式的出队列.
  • BlockingQueue 也有 offer, poll, peek 等方法, 但是这些方法不带有阻塞特性.
BlockingQueue<String> queue = new LinkedBlockingQueue<>();
// 入队列
queue.put("abc");
// 出队列. 如果没有 put 直接 take, 就会阻塞. 
String elem = queue.take();

//生产者消费者模型
public static void main(String[] args) throws InterruptedException {
    BlockingQueue<Integer> blockingQueue = new LinkedBlockingQueue<Integer>();
    Thread customer = new Thread(() -> {
        while (true) {
            try {
                int value = blockingQueue.take();
                System.out.println("消费元素: " + value);
           } catch (InterruptedException e) {
                e.printStackTrace();
           }
       }
   }, "消费者");


    customer.start();
    Thread producer = new Thread(() -> {
        Random random = new Random();
        while (true) {
            try {
                int num = random.nextInt(1000);
                System.out.println("生产元素: " + num);
                blockingQueue.put(num);
                Thread.sleep(1000);
           } catch (InterruptedException e) {
                e.printStackTrace();
           }
       }
   }, "生产者");
    producer.start();
    customer.join();
    producer.join();
}

5.1.3 消息队列

本质上就是一个阻塞队列,在此基础上为放入阻塞队列的消息打一个标签. 

实现了分组的作用

 5.1.4 消息队列的作用

1.解耦:以下通过画图解释 更加易懂

 

 2. 削峰填谷

峰与谷指消息的密集程度

举个栗子:三峡大坝  

汛期:起到蓄水的功能,防止下游遭受洪峰的冲击  削峰

旱期:可以把存的水源源不断地向下游排放   填谷

再比如说 在工程环境中的应用  微博出现热点事件时

 

 

 3. 异步:发出请求之后,自己去干别的事情,有响应时会接受到通从而处理响应

演示JDK中提供的阻塞队列:

public static void main(String[] args) throws InterruptedException {
        // 定义一个阻塞队列
        BlockingQueue<Integer> queue = new LinkedBlockingQueue<>(3);
        // 往队列中写入元素
        queue.put(1);
        queue.put(2);
        queue.put(3);
        System.out.println("已经插入了三个元素");
        System.out.println(queue);
//        queue.put(4);
//        System.out.println("已经插入了四个元素");

        System.out.println("开始获取元素");
        // 阻塞队列中获取元素使用take,会产生阻塞效果
        System.out.println(queue.take());
        System.out.println(queue.take());
        System.out.println(queue.take());
        System.out.println("已经获取了三个元素");
        System.out.println(queue.take());
        System.out.println("已经获取了四个元素");

        System.out.println(queue);


    }
}

 

 5.2 实现一个阻塞队列

  • 之前实现一个普通队列,底层运用到了两种数据结构,一个是链表,一个是循环数组
  • 阻塞队列就是在普通的队列上加入了阻塞等待的操作

代码实现:

public class MyBlockingQueue {
    // 定义一个保存元素的数组
    private int[] elementData = new int[100];
    // 定义队首下标
    private volatile int head;
    // 定义队尾下标
    private volatile int tail;
    // 定义一个有效元素的个数
    private volatile int size;

    /**
     * 插入一个元素
     * @param value
     */
    public void put (int value) throws InterruptedException {
        // 根所修改共享变量的范围加锁,锁对象this即可
        synchronized (this) {
            // 判断数据是不是已经满了
            while (size >= elementData.length) {
                // 阻塞等待
                this.wait();
            }
            // 向队尾去插入元素
            elementData[tail] = value;
            // 移动队尾下标
            tail++;
            // 修正队尾下标
            if (tail >= elementData.length) {
                tail = 0;
            }
            // 修改有效元素的个数
            size++;
            // 做唤醒操作
            this.notifyAll();
        }
    }

    /**
     * 获取一个元素
     * @return
     */
    public int take() throws InterruptedException {
        // 根所修改共享变量的范围加锁
        // 锁对象this即可
        synchronized (this) {
            // 判断队列是否为空
            while (size <= 0) {
                this.wait();
            }
            // 从队首出队
            int value = elementData[head];
            // 移动队首下标
            head++;
            // 修改队首下标
            if (head >= elementData.length) {
                head = 0;
            }
            // 修改有效元素的个数
            size--;
            // 唤醒操作
            this.notifyAll();
            // 返回队首元素
            return value;
        }
    }
}

图文分析:

 分析后发现整个方法都存在修改共享变量的操作,所以给整个方法加锁

 确定唤醒时机

 测试结果符合预期

虚假唤醒

 线程也可以在没有通知、中断或超时的情况下唤醒,即所谓的虚假唤醒。虽然这种情况在现实生活中很少发生,但应用程序必须通过测试应该导致线程被唤醒的条件来防止这种情况,如果条件不满足,则继续等待。换句话说,等待应该总是出现在循环中。

简而言之第一次满足的条件,线程进入阻塞状态,那被唤醒之后,这期间会发生很多事情,有一种可能是被唤醒之后等待条件依然成立,答案是肯定的,所以需要再次检查等待条件。

所以代码中所有需要条件判断的wait,强烈建议加入到while循环


六、线程池

6.1 线程池是什么?

JDBC编程中,通过DataSourse获取Connection的时候就已经用到了的概念

 当JAVA程序需要数据库连接的时候,就从池子中拿一个空闲的连接对象给JAVA程序,JAVA程序用完了连接之后就会返回给连接池,线程池就是在池子里放的线程本身,当程序启动的时候就创建出若干个线程,如果有任务就处理,没有任务就阻塞等待.

举个栗子:

在学校附近新开了一家快递店老板很精明,想到一个与众不同的方法来经营,店里没有雇人,而是每次有业务来了,就现场找一名同学把快递送了,然后解雇同学,这个类比我们平时来一个任务,起一个线程进行处理的模式。

很快,老板发现问题来了,每次招聘和解雇同学的成本还是非常高的,老板还是很善于变通的,知道为什么大家都要雇人了,所以指定每一个指标,公司业务人员会扩张到三个人,但是还是随着业务逐步雇人,于是再有业务来了,老板就看,如果现在公司还没三个人,就雇一个人去送快递,否则只是把业务放在一个本本上,等着三个快递人员空闲的时候去处理。这就是我们要带出的线程池的模式线程池,最大的好处就是减少每次启动销毁线程的损耗。

 

6.2 怎么使用线程池?

6.2.1 JDK给我们提供了一些方法来创建线程池(不建议使用)

        // 1. 用来处理大量短时间工作任务的线程池,如果池中没有可用的线程将创建新的线程,如果线程空闲60秒将收回并移出缓存
        ExecutorService cachedThreadPool = Executors.newCachedThreadPool();
        // 2. 创建一个操作无界队列且固定大小线程池 (3)创建线程时,池中包含了3条线程
          (无界队列:对于队列中的元素不加个数,可能会出现内存被消耗殆尽的情况)
    ExecutorService fixedThreadPool = Executors.newFixedThreadPool(3);
        // 3. 创建一个操作无界队列且只有一个工作线程的线程池
        ExecutorService singleThreadExecutor = Executors.newSingleThreadExecutor();
        // 4. 创建一个单线程执行器,可以在给定时间后执行或定期执行。
        ScheduledExecutorService singleThreadScheduledExecutor = Executors.newSingleThreadScheduledExecutor();
        // 5. 创建一个指定大小的线程池,可以在给定时间后执行或定期执行。
        ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(3);
        // 6. 创建一个指定大小(不传入参数,为当前机器CPU核心数)的线程池,并行地处理任务,不保证处理顺序
        Executors.newWorkStealingPool();

6.2.2 工厂模式

解决构造方法创建对象的不足

举个栗子:


class Student {


    private int id;
    private int age;
    private String name;// 通过id和name属性来构造一个学生对象
    public Student (int id , String name) {
       this.id = id;
     this.name = name;

    }

    // 通过age 和 name 属性来构造一个学生对象
    public Student (int age , String name) {
        this.age = age;
      this.name = name;
    }

由于重载过程参数列表相同而报错!

 // 通过id和name属性来构造一个学生对象
    public static Student createByIdAndName (int id, String name) {
        Student student = new Student();
        student.setId(id);
        student.setName(name);
        return student;
    }

    // 通过age 和 name 属性来构造一个学生对象
    public static Student createByAgeAndName (int age, String name) {
        Student student = new Student();
        student.setAge(age);
        student.setName(name);
        return student;
    }

工厂模式就是:传来什么样的数据,按照工厂方法里的逻辑返回什么对象

6.3 自定义一个线程池

1.可以考虑提交任务到线程池,那么就会有一种数据结构来保存我们提交的任务,

可以考虑用阻塞队列来保存任务. 

2.创建线程池是需要指定初始化线程数据,这些线程不停的扫描阻塞队列,如果有任务就立即执行 

可以考虑使用线程池对象的构造方法,接受要创建线程的数据,并在构造方法中完成线程的创建

public class Demo03_ThreadPool_Use {
    public static void main(String[] args) throws InterruptedException {
        // 创建一个线程池
        ExecutorService threadPool = Executors.newFixedThreadPool(3);
        // 提交任务到线程池
        for (int i = 0; i < 10; i++) {
            int taskId = i;
            threadPool.submit(() -> {
                System.out.println("我是任务 " + taskId + ", " + Thread.currentThread().getName());
            });
        }

        // 模拟等待任务
        TimeUnit.SECONDS.sleep(5);
        System.out.println("第二阶段开始");

        // 提交任务到线程池
        for (int i = 10; i < 20; i++) {
            int taskId = i;
            threadPool.submit(() -> {
                System.out.println("我是任务 " + taskId + ", " + Thread.currentThread().getName());
            });
        }



    }
}

实现过程:

public class MyThreadPool {
    // 1. 定义一个阻塞队列来保存我们的任务
    BlockingQueue<Runnable> queue = new LinkedBlockingQueue<>(3);

    // 2. 对外提供一个方法,用来往队列中提交任务
    public void submit (Runnable task) throws InterruptedException {
        queue.put(task);
    }

    // 3. 构造方法
    public MyThreadPool (int capacity) {
        if (capacity <= 0) {
            throw new RuntimeException("线程数量不能小于0.");
        }
        // 完成线程的创建,扫描队列,取出任务并执行
        for (int i = 0; i < capacity; i++) {
            // 创建线程
            Thread thread = new Thread(() -> {
                while (true) {
                    try {
                        // 取出任务(扫描队列的过程)
                        Runnable take = queue.take();
                        // 执行任务
                        take.run();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            });
            // 启动线程
            thread.start();

        }
    }

}

6.3.1 为什么不推荐使用系统自带的线程池?

通过工厂方法获取的线程池,最终都是ThreadPoolExecutor类的对象。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/517611.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

5G通信-帧结构及RE和RB

一.NR的帧结构&#xff08;可以参考38.211&#xff09; NR的时间单位由大到小依次为帧&#xff0c;子帧&#xff0c;时隙&#xff0c;符号。 其中一帧数据&#xff08;10ms&#xff09;由10个子帧组成&#xff0c; 一个子帧&#xff08;1ms&#xff09;由若干个时隙组成&#…

3个方案,绕过微信小程序官方审核

描述 虽然绕过官方审核&#xff0c;是不推荐的行为&#xff0c;但是官方的做法有点难以接受。偶尔会碰见奇葩理由被拒绝。例如&#xff1a;类目不对、功能过于简单、涉及金融(其实没有)、涉及官方素材等等。之前小程序不多的时候&#xff0c;你总是求着我加入开发&#xff1b;…

MYSQL相关之不常见变量、排序函数、JDBC数据库与Java连接

用户自定义变量 局部变量--->只在当前begin/end代码块中有效 sql 复制代码 create procedure add ( in a int, in b int ) begin declare c int default 0; set c a b; select c as c; end; 2.用户变量--->在客户端链接到数据库实例整个过程中用户变量都是有效的。…

商户查询的缓存——缓存穿透

4.缓存穿透 缓存穿透是指客户端请求的数据在缓存中和数据库中都不存在&#xff0c;这样缓存永远不会生效&#xff0c;这些请求都会打到数据库 解决方案&#xff1a; 1.缓存空对象&#xff08;简单粗暴&#xff09; 2.布隆过滤 代码实现&#xff1a; Autowired private String…

在米文动力 EVO Orin 设备 Jetson AGX Orin 下配置 YOLOV5 的环境说明

声明&#xff1a;本博文的配置大部分参考 Jetson AGX Orin安装Anaconda、Cuda、Cudnn、Pytorch、Tensorrt最全教程 1 前言 本文设备内置的模组是 Nvidia Jetson AGX Orin 32GB&#xff0c;算力达到 200 TOPS&#xff0c;aarch64 架构 本文主要参考《Jetson AGX Orin安装Anac…

发明专利公开 -- CSS动画精准实现时钟

上一篇【setTimeout不准时&#xff0c;CSS精准实现计时器功能】的博文&#xff0c;最后提到了通过 CSS 动画实现计时器的方式。 本文详情描述如何通过 CSS 完整实现时钟效果&#xff0c;这也是团队 21 年专利的一项内容&#xff08;专利公布号&#xff1a;CN114003087A&#x…

Docker安装CentOS7

Docker 分为 CE 和 EE 两大版本。CE 即社区版&#xff08;免费&#xff0c;支持周期 7 个月&#xff09;&#xff0c;EE 即企业版&#xff0c;强调安全&#xff0c;付费使用&#xff0c;支持周期 24 个月。 本教程是关于在CentOS上安装社区版Docker的方法&#xff08;Docker C…

SCTP:记录一次Diameter消息卡顿延迟问题

SCTP&#xff1a;记录一次Diameter消息卡顿延迟问题 1、背景 客户端&#xff1a;5个接口机 服务端&#xff1a;2个模拟器&#xff08;模拟HSS&#xff0c;是一个Diameter服务端&#xff09; 5个客户端的地址是&#xff1a; 10.212.27.2910.212.27.5510.212.24.1710.212.24.1…

js编译、执行上下文、作用域链

参考资料 极客时间课程《浏览器工作原理与实践》 – 李兵 《你不知道的JavaScript》-- Kyle Simpson ES5.1规范&#xff1a;https://262.ecma-international.org/5.1/#sec-10.3 ES6规范&#xff1a;https://262.ecma-international.org/6.0/#sec-executable-code-and-execu…

Redis 五大基本数据类型常见命令

一、redis中的常见数据结构 Redis共有5种常见数据结构&#xff0c;分别字符串&#xff08;STRING)、列表&#xff08;LIST&#xff09;、集合&#xff08;SET)、散列&#xff08;HASH&#xff09;、有序集合&#xff08;ZSET)。 二、redis中字符串(String)介绍 String 类型是…

小红书内容种草怎么玩,koc铺量原则

一直以来&#xff0c;小红书平台都以其强大的种草力&#xff0c;而备受品牌关注。许多初创品牌更是将平台看做抢占市场的前沿阵地。那么小红书内容种草怎么玩&#xff0c;koc铺量原则是什么呢?今天来为大家详细介绍一下。 一、什么是koc铺量 要做好新品预热的koc铺量&#xff…

存bean和取bean

准备工作存bean获取bean三种方式 准备工作 bean:一个对象在多个地方使用。 spring和spring boot&#xff1a;spring和spring boot项目&#xff1b;spring相当于老版本 spring boot本质还是spring项目&#xff1b;为了方便spring项目的搭建&#xff1b;操作起来更加简单 spring…

有效和无效的帮助中心区别在哪?如何设计有效的帮助中心?

帮助中心就是一个丰富的知识库&#xff0c;可以对企业的潜在客户进行引导。不仅能够提升用户的使用体验还能为企业塑造更加专业的品牌形象&#xff0c;在使用过程中为用户提供帮助。帮助中心的目的就是为了解决用户在使用过程中遇到的困难&#xff0c;同时为用户的使用提供引导…

DC-9通关详解

信息收集 漏洞发现 result.php处存在sql注入 sqlmap跑信息 python sqlmap.py -u http://192.168.45.146/results.php --data search1 -D users -T UserDetails --dump 拿了几个尝试登录都无效 ssh尝试登录直接拒绝了 再看Staff表 查哈希 进后台 多了一个添加记录的功能 没啥…

数据结构(二叉树)

文章目录 一、树的基础概念1.1 树型结构1.2 树型的概念 二、二叉树2.1 概念 性质2.2 二叉树的存储2.2 二叉树的基本操作&#xff08;1&#xff09;遍历&#xff08;2&#xff09;其他 2.3 二叉树练习 一、树的基础概念 1.1 树型结构 树是一种非线性的数据结构&#xff0c;它…

ChatGPT 由0到1接入 Siri

ChatGPT 由0到1接入 Siri ChatGPT 由0到1接入 Siri第一步:获取 OpenAPI 的 Key第二步:制作快捷指令本教程收集于: AIGC从入门到精通教程 ChatGPT 由0到1接入 Siri 分享如何将 GPT 应用集成到苹果手机的 Siri 中 (当然手机是需要魔法(TZ)的) 第一步:获取 OpenAPI 的…

网络安全可以从事哪些岗位?岗位职责是什么?

伴随着社会的发展&#xff0c;网络安全被列为国家安全战略的一部分&#xff0c;因此越来越多的行业开始迫切需要网安人员&#xff0c;也有不少人转行学习网络安全。那么网络安全可以从事哪些岗位?岗位职责是什么?相信很多人都不太了解&#xff0c;我们一起来看看吧。 1、安全…

电阻阻值读取方法、电容容值的读取方法

电阻、电容的数值读取方法 文章目录 电阻、电容的数值读取方法前言1、电阻读数1.1 贴片电阻1.2.直插色环电阻 2、电容读数2.1 电容单位换算2.2 电容读数方法 前言 现在随着电子产品的不断升级优化&#xff0c;做到体积越来越小了&#xff0c;以前发现还是用得很多直插电阻和一…

百年不用了,今天拾起来 sort() 排序

简单赘述一下需求。 原本前端调用后端接口是自带排序功能的&#xff0c;一般是按照创建单据的时间&#xff0c;后端会处理好返回给我们。 但是有时候有特别的限制&#xff0c;需要前端自行处理排序展示。 如上图所示&#xff0c; 列表和列表扩展行均要根据我们新增或编辑的时候…

ATTCK v13版本战术介绍——防御规避(五)

一、引言 在前几期文章中我们介绍了ATT&CK中侦察、资源开发、初始访问、执行、持久化、提权战术理论知识及实战研究、部分防御规避战术&#xff0c;本期我们为大家介绍ATT&CK 14项战术中防御规避战术第25-30种子技术&#xff0c;后续会介绍防御规避其他子技术&#xf…