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

 五、阻塞队列

目录

 五、阻塞队列

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类的对象。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

多媒体基础

第九章、多媒体基础 1、多媒体技术基本概念 1.1、音频相关概念 超声波的频率通常在20千赫兹以上&#xff0c;无法被人类的耳朵听到&#xff0c;常用于医疗诊断、非破坏性材料测试、清洗、测量等领域 次声波的频率通常在20赫兹以下&#xff0c;同样无法被人类的耳朵听到&…

数据库缓存服务——NoSQL之Redis配置与优化

一、缓存概念 缓存是为了调节速度不一致的两个或多个不同的物质的速度&#xff0c;在中间对速度较慢的一方起到加速作用&#xff0c;比如CPU的一级、二级缓存是保存了CPU最近经常访问的数据&#xff0c;内存是保存CPU经常访问硬盘的数据&#xff0c;而且硬盘也有大小不一的缓存…

11个超好用的SVG编辑工具

SVG的优势在于SVG图像可以更加灵活&#xff0c;自由收缩放大而不影响图片的质量&#xff0c;一个合适的SVG编辑工具能够让你的设计事半功倍&#xff0c;下面就一起来看看这些冷门软件好用在哪里。这11个超好用的SVG编辑工具依次为&#xff1a;即时设计、Justinmind、Sketsa SVG…

MATLAB绘制动画(二)擦除动画

如果我们在绘制图形之后将原有的图形擦除&#xff0c;并重新绘制&#xff0c;看上去就像动画了 示例: t 0; m [sin(t);cos(t)]; p plot(t,m,EraseMode,background,MarkerSize,5); x -1.5*pi; axis([x x2*pi -1.5 1.5]); grid onfor i 1:100t [t 0.1*i];m [m [sin(0.1*i…

BitKeep逆势崛起:千万用户的信任,终点还未到来

在全球范围内&#xff0c;BitKeep钱包如今已拥有超过千万忠实用户。 当我得知这一令人震撼的数字时&#xff0c;既感到惊讶&#xff0c;同时也觉得这是意料之中的事情。几年来关注BitKeep的发展历程&#xff0c;我深切地感受到了这家公司的蓬勃壮大。回顾2018年他们发布的第一个…

JVM 堆

堆的核心概述 一个 JVM 实例只存在一个堆内存&#xff0c;堆也是 Java 内存管理的核心区域Java 堆区在 JVM 启动的时候即被创建&#xff0c;其空间大小也就确定了。是 JVM 管理的最大一块内存空间堆可以处于物理上不连续的内存空间中&#xff0c;但是在逻辑上它应该被视为连续…

airserver7.2.7最新中文版下载及功能介绍

最近开会打算把手机投屏到自己的Mac上演示用&#xff0c;于是就打算用下听了很久好用但是一值没有使用的AirServer!十分简单的操作就可以完美的投屏到Mac电脑&#xff0c;而且不用像Mac自带的QuickTime用线连接手机!它可以把AirPlay / AirTunes上的音频、视频、照片、幻灯片还有…

对称算法模式-GCM(Galois/Counter Mode)

以下内容来自《NIST Special Publication 800-38D November, 2007》- Recommendation for Block Cipher Modes of Operation: Galois/Counter Mode (GCM) and GMAC。 链接在此 AES Galois/Counter Mode 1. 加密步骤 2. 解密步骤 3. GCTR函数 4. GHASH函数 5. 块数据乘法 6. C…

大势智慧软硬件技术答疑第三期

1.重建大师6.0试用版&#xff0c;怎么导出DOM、DEM&#xff1f; 答&#xff1a;需要先生成三维模型&#xff0c;然后再提交产品选择DOM和DEM。 2.麻烦问下&#xff0c;修模出来贴的纹理图片&#xff0c;导出osgb后再打开就模糊了是什么情况&#xff1f; 答&#xff1a;拿高清…

【学习笔记-myabtis】使用mybtis对接pgsql的postgis插件,获取地理字段Geometry信息

使用mybtis对接pgsql的postgis插件&#xff0c;获取地理字段geometry信息 参考资料&#xff1a; Mybatis 自定义TypeHandler - 邓维-java - 博客园 1、如何使用typehandler ​ 相信大家用Mybatis这个框架至少一年以上了吧&#xff0c;有没有思考过这样一个问题&#xff1a;数据…

Hello, Mojo——首次体验Mojo语言

Hello, Mojo——首次体验Mojo语言 文章目录 Hello, Mojo——首次体验Mojo语言一、前言二、Mojo有哪些独特的功能使它不同于Python&#xff1f;三、可以在 Mojo 中导入和使用的 Python 哪些包&#xff1f;四、为什么参数化在 Mojo 中对于使用 SIMD 类型和硬件加速器很重要&#…

[Orillusion]-使用 -windwos-4行命令

前两天看了webgpu的开源库Orillusion | 专业 WebGPU 引擎 Orillusion感觉很不错的样子&#xff0c;准备试一下。因为都是做OpenGL和windows桌面端。 web有点小陌生&#xff0c;记录一下。 准备&#xff1a; Google Chrome Canary 最新版&#xff0c;老版本有问题 nodejs 版…

k8s架构了解

Kubernetes(k8s)是用于自动部署、扩展和管理“容器化应用程序”的开源系统 k8s由control plane以及cluster nodes构成 control plane control plane是维护所有k8s对象记录的系统&#xff0c;持续管理着对象状态&#xff0c;并且对集群的变化做出响应&#xff0c;并使状态匹…

matlab实验三程序设计与优化

学聪明点&#xff0c;自己改&#xff0c;别把我卖了 一、实验目的及要求 一、实验的目的与要求 1、掌握 MATLAB的函数 2、掌握 MATLAB的程序流 3、掌握 MATLAB脚本和函数文件的编写 4、熟悉基于矩阵的程序设计与优化 二、实验原理 1、MATLAB的M文件&#xff1a;脚本文件与函数…

MMM(Master-Master replication manager for MySQL)

MMM&#xff08;Master-Master replication manager for MySQL&#xff0c;MySQL主主复制管理器&#xff09; 是一套支持双主故障切换和双主日常管理的脚本程序。MMM 使用 Perl 语言开发&#xff0c;主要用来监控和管理 MySQL Master-Master &#xff08;双主&#xff09;复制&…

matlabR2021b启动很慢和初始化时间很长解决

工具&#xff1a;MatlabR2021b。 问题记录&#xff0c;在网上下载安装包后&#xff0c;安装后&#xff0c;发现软件启动时间很长。进入界面后软件需要较长时间的初始化。才能就绪。 查询原因为软件需要在启动是查询licence。 首先在安装文件夹中启动Activate MATLAB R2021b。…

图解LeetCode——240. 搜索二维矩阵 II

一、题目 编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target 。该矩阵具有以下特性&#xff1a; 每行的元素从左到右升序排列。 每列的元素从上到下升序排列。 二、示例 2.1> 示例 1&#xff1a; 【输入】matrix [[1,4,7,11,15],[2,5,8,12,19],[3,6,9,1…

leaflet根据坐标点设置多边形,生成geojson文件,计算面积值(133)

第133个 点击查看专栏目录 本示例的目的是介绍演示如何在vue+leaflet中根据坐标点设置多边形,通过.toGeoJSON() 来生成geojson文件,通过turf.area来计算面积值。 直接复制下面的 vue+leaflet源代码,操作2分钟即可运行实现效果 文章目录 示例效果配置方式示例源代码(共123…

Java基础之ConcurrentHashMap答非所问

ConcurrentHashMap的数据结构是什么&#xff1f; ConcurrentHashMap仅仅是HashMap的线程安全版本&#xff0c;针对HashMap的线程安全优化&#xff0c;所以HashMap有的特点ConcurrentHashMap同意具有&#xff0c; ConcurrentHashMap的数据结构跟HashMap是一样的。 在JDK7版本使用…

QTableView编程——Model/View架构(单元格随意拖拽交换)

QTableView编程——Model/View架构 基础知识 添加表头 //准备数据模型QStandardItemModel *student_model new QStandardItemModel();student_model->setHorizontalHeaderItem(0, new QStandardItem(QObject::tr("Name")));student_model->setHorizontalHea…