【线程】Java多线程代码案例(1)
【线程】Java多线程代码案例(1)
- 一、“单例模式” 的实现
- 1.1“饿汉模式”
- 1.2 “懒汉模式”
- 1.3 线程安全问题
- 二、“阻塞队列”的实现
- 2.1阻塞队列
- 2.2生产者消费者模型
- 2.3 阻塞队列的实现
- 2.4 再谈生产者消费者模型
一、“单例模式” 的实现
“单例模式”即在一个Java 进程中,要求指定的类,只能有一个实例。通过一些特殊的技巧,来确保这里的实例不会有多个。
1.1“饿汉模式”
class Singleton{//在这个类被加载的时候,初始化这个静态成员private static Singleton instance=new Singleton();public static Singleton getInstance() {return instance;}//构造方法私有,在类外部不允许new新的实例private Singleton(){}
}
- 在类外不允许创建新的实例:
- instance实例时Singleton类的静态成员变量,在内存中只有一份存储空间,是唯一的。
1.2 “懒汉模式”
“懒汉模式”创建实例的时机会相对晚一些,只有到第一次使用的时候,才会创建实例。
class Singleton{private static Singleton instance=null;public static Singleton getInstance() {if (instance==null) {instance=new Singleton();}return instance;}//构造方法私有,在类外部不允许new新的实例private Singleton(){}}
同样,在类外不允许创建新的实例。但是在类内创建实例的时机会相对晚一些。
1.3 线程安全问题
再次观察这两份代码,是否会存在线程安全问题呢?
其实在“饿汉模式”下,并不存在线程安全的问题,因为此处在加载类的时候,就创建了唯一的实例,后续使用只涉及到了“读取”这个操作;
而在“懒汉模式”下,就会存在一些线程安全问题,我们接下来仔细分析此处可能存在的线程安全问题:
- 由“随机调度”引起的线程安全问题
因为在“懒汉模式”下,既涉及到了读的操作,也涉及到了写的操作。如果在多线程中出现这样的执行顺序,就会创建出多个实例,在我们要求“单例模式”的初心下,就出现了问题。
这样的问题,也很容易解决,无非是通过加锁将读写操作打包到一起,使其成为“原子性”的操作。
public static Singleton getInstance() {synchronized (locker) {if (instance == null) {instance = new Singleton();}}return instance;}
这样的话,instance == null
和 instance=new Singleton()
这两个操作就被打包到一起,就不会出现这样的问题了。但是,这份代码仍有一些问题。
- 效率上的问题
上述线程安全问题,其实只会在第一次创建实例的时候有可能会出现。也就是在整个使用过程中,只有第一次需要加锁,而我们现在的策略是每次读取instance实例都要加锁,就会造成效率低下。
public static Singleton getInstance() {if(instance==null) {synchronized (locker) {if (instance == null) {instance = new Singleton();}}}return instance;}
对于这样的问题,我们可以在外层再加一层判定,只在第一次创建实例的时候才加锁。
这样的代码,第一次见可能还是会有些疑惑,明明是两个一样的判定条件,怎么就需要两次?
首先我们分析,外面这个判定不要了可以吗?不行,这会影响效率。
那么,里面这个判定不要了可以吗?也不可以,如果里面这层判定没有,本质上instance == null
和instance=new Singleton()
这两步操作没有被打包到一起,仍会存在线程安全的问题。
那么,两层判定都不能去掉,究竟有何玄机呢?
这其实是因为加锁会引起阻塞,也就是意味着这两个判定执行的时间是不相同的,在阻塞的这段时间内,别的线程可能完成了一些操作(比如此处的新建实例),使得判定不再成立。
解决了效率上的问题,还可能会存在一些问题,我们继续分析。
- 由“系统优化”引起的指令重排序问题
instance = new Singleton();
这条代码,可简单理解为由三个步骤组成:
(1)申请一段内存空间;
(2)在这个内存上调用构造方法,创建出这个实例;
(3)把这个内存地址赋值给Instance。
正常情况下,系统是按1、2、3的步骤执行的,也有可能会优化为1、3、2的执行顺序。如果在单线程中,1、3、2的执行顺序不会有什么问题,但在多线程中,未必如此。
如果是这样的执行顺序,那t2线程返回的instance
实例,就是一个未初始化的“全0”的值,就会出现问题。
那么这样的问题,如何解决?
依靠于Java中提供的volatile
关键字,我们之前提到它可以强制系统从内存中读取数据、避免出现“内存可见性问题”。此处是volatile
的第二个功能,禁止指令重排序。
private volatile static Singleton instance=null;
到此,这段代码就实现了既高效,且避免了线程安全的问题。完整代码如下:
class Singleton{private volatile static Singleton instance=null;private static Object locker;public static Singleton getInstance() {if(instance==null) {synchronized (locker) {if (instance == null) {instance = new Singleton();}}}return instance;}//构造方法私有,在类外部不允许new新的实例private Singleton(){}}
二、“阻塞队列”的实现
2.1阻塞队列
阻塞队列,是基于普通队列做出的扩展(先进先出),优势在于:
- 线程安全
- 具有阻塞特性
(1)如果一个队列已经满了,此时入队列就会阻塞,直到队列不满;
(2)如果一个队列已经空了,此时出队列就会阻塞,直到队列不空。
在 Java 标准库中内置了阻塞队列. 如果我们需要在⼀些程序中使⽤阻塞队列, 直接使⽤标准库中的即可.
- BlockingQueue 是⼀个接⼝. 由
ArrayBlockingQueue
、LinkedBlockingQueue
、PriorityBlockingQueue
实现。 - put ⽅法⽤于阻塞式的⼊队列, take ⽤于阻塞式的出队列.
- BlockingQueue 也有 offer, poll, peek 等⽅法, 但是这些⽅法不带有阻塞特性.
2.2生产者消费者模型
引入生产者消费者模型,
- 一方面是为了“解耦合”,也就是说比如原来是生产者和消费者直接交互,那么如果消费者变更、或者生产者变更,我们的代码就需要有比较大的改动,现在统一面向缓冲区(阻塞队列),就可以很好的“解耦合”。
- 另一方面是“削峰填谷”,通过这个阻塞队列可以很好的控制消费者读取信息的速率。避免短时间内出现大量数据,引起问题。
2.3 阻塞队列的实现
实现阻塞队列,实质上就是实现三个方面:
- 先实现普通队列
class MyBlockingQueue{private String[] elems=null;private int head=0;private int tail=0;private int size=0;public MyBlockingQueue(int capacity){elems=new String[capacity];}public void put(String elem){if(size >= elems.length){//队列满了,此处要能够阻塞}elems[tail]=elem;tail++;if(tail>= elems.length){tail=0;}size++;}public String take(){String elem=null;if(size == 0){//队列空了,要能阻塞}elem=elems[head];head++;if(head >= elems.length){head=0;}size--;return elem;}}
- 再加上线程安全
我们要求入队列和出队列、多次入队列、多次出队列不能并行执行,就要通过加锁实现。那锁都要包含那些操作呢?
那么这个判断队列是否满了的操作是否需要包裹?
试想这样的执行顺序,如果是入队列最后一个元素,就会出现问题。所以加锁应该包含判断队列是否为满的操作。
public void put(String elem) {synchronized (locker) {if (size >= elems.length) {//队列满了,此处要能够阻塞}elems[tail] = elem;tail++;if (tail >= elems.length) {tail = 0;}size++;}}public String take() {synchronized (locker) {String elem = null;if (size == 0) {//如果队列为空,实现阻塞}elem = elems[head];head++;if (head >= elems.length) {head = 0;}size--;return elem;}}
- 再加上阻塞功能
我们通过wait()和notify()实现。
public void put(String elem) throws InterruptedException {synchronized (locker) {if (size >= elems.length) {//队列满了,此处要能够阻塞locker.wait();}elems[tail] = elem;tail++;if (tail >= elems.length) {tail = 0;}size++;locker.notify();}}public String take() throws InterruptedException {synchronized (locker) {String elem = null;if (size == 0) {locker.wait();}elem = elems[head];head++;if (head >= elems.length) {head = 0;}size--;locker.notify();return elem;}}
看到这样的代码,想必大家心中会有一个问题,你这一个锁里又wait()、又notify()的,这能行吗?不是自己把自己唤醒了吗?
但实际上这种问题还真不会出现。
真实情况应该是这样:
虽然我们刚才所说的这种问题不会出现,但仍有可能出现“连环唤醒”问题:
这里我们假设队列空了进入阻塞,我们通过put()加入一个元素后,这里随机唤醒一个线程进行take()操作,这个线程执行过程中,又把另一个线程唤醒进行take(),此时就会出现问题。
如何解决?
关键在于此处的 if 只判定一次。第一个线程被唤醒取走一个元素、并将第二个线程唤醒后,第二个线程如果能在判定1次队列是否为空,此时就不会出现问题。
这也很容易实现,我们只需要将 if 改成 while 即可。
2.4 再谈生产者消费者模型
我们可以通过通过控制入队或者出队的速度,来模拟这一过程。
//生产者
Thread t1=new Thread(()->{int n=1;while(true){try {queue.put(n+"");System.out.println("生产元素 "+n);n++;//2. 生产慢消费快Thread.sleep(500);} catch (InterruptedException e) {throw new RuntimeException(e);}}});
//消费者Thread t2=new Thread(()->{while(true){try {String n=queue.take();System.out.println("消费元素 "+n);//1. 生产块消费慢Thread.sleep(500);} catch (InterruptedException e) {throw new RuntimeException(e);}}});
-
生产快消费慢
-
生产慢消费快
完整代码:
class MyBlockingQueue{private String[] elems=null;private int head=0;private int tail=0;private int size=0;private static Object locker=new Object();public MyBlockingQueue(int capacity){elems=new String[capacity];}public void put(String elem) throws InterruptedException {synchronized (locker) {while(size >= elems.length) {//队列满了,此处要能够阻塞locker.wait();}elems[tail] = elem;tail++;if (tail >= elems.length) {tail = 0;}size++;locker.notify();}}public String take() throws InterruptedException {synchronized (locker) {String elem = null;while(size == 0) {locker.wait();}elem = elems[head];head++;if (head >= elems.length) {head = 0;}size--;locker.notify();return elem;}}}public class ThreadDemo4 {public static void main(String[] args) {MyBlockingQueue queue=new MyBlockingQueue(5);//生产者Thread t1=new Thread(()->{try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}int n=1;while(true){try {queue.put(n+"");System.out.println("生产元素 "+n);n++;//Thread.sleep(500);} catch (InterruptedException e) {throw new RuntimeException(e);}}});//消费者Thread t2=new Thread(()->{while(true){try {String n=queue.take();System.out.println("消费元素 "+n);Thread.sleep(500);} catch (InterruptedException e) {throw new RuntimeException(e);}}});t1.start();t2.start();}
}
相关文章:
【线程】Java多线程代码案例(1)
【线程】Java多线程代码案例(1) 一、“单例模式” 的实现1.1“饿汉模式”1.2 “懒汉模式”1.3 线程安全问题 二、“阻塞队列”的实现2.1阻塞队列2.2生产者消费者模型2.3 阻塞队列的实现2.4 再谈生产者消费者模型 一、“单例模式” 的实现 “单例模式”即…...
go使用mysql实现增删改查操作
1、安装MySQL驱动 go get -u github.com/go-sql-driver/mysql2、go连接MySQL import ("database/sql""log"_ "github.com/go-sql-driver/mysql" // 导入 mysql 驱动 )type Users struct {ID intName stringEmail string }var db *sql.DBfu…...
【Rust】unsafe rust入门
这篇文章简单介绍下unsafe rust的几个要点 1. 解引用裸指针 裸指针其实就是C或者说C的指针,与C的指针不同的是,Rust的裸指针还是要分为可变和不可变,*const T 和 *mut T: 基于引用创建裸指针 let mut num 5;let r1 &num …...
dpwwn02靶场
靶机下载地址:https://download.vulnhub.com/dpwwn/dpwwn-02.zip 信息收集 ip add 查看kali Linux虚拟机的IP为:10.10.10.128 https://vulnhub.com/entry/dpwwn-2,343/中查看靶机的信息,IP固定为10.10.10.10 所以kali Linux添加仅主机网卡…...
K8S疑难概念理解——Pod,应该以哪种Kind来部署应用,为什么不直接Pod这种kind?
文章目录 一、Pod概念深度理解,为什么一般不直接以kindPod资源类型来部署应用?二、究竟应该以哪种资源类型来部署应用 一、Pod概念深度理解,为什么一般不直接以kindPod资源类型来部署应用? Pod是Kubernetes中的最小部署单元,可以包含一个或…...
LabVIEW进行仪器串行通信与模拟信号采集的比较
在现代测试、测量和控制系统中,设备通常采用两种主要方式与计算机进行交互:一种是通过数字通信接口(如RS-232、RS-485、GPIB等),另一种是通过模拟信号(电压、电流)进行数据输出。每种方式具有其…...
D81【 python 接口自动化学习】- python基础之HTTP
day81 requests请求session用法 学习日期:20241127 学习目标:http定义及实战 -- requests请求session用法 学习笔记: requests请求session用法 import requests# 创建一个会话 reqrequests.session() url "http://sellshop.5istud…...
白鹿 Hands-on:消除冷启动——基于 Amazon Lambda SnapStart 轻松打造 Serverless Web 应用(二)
文章目录 前言一、前文回顾二、在 Lambda 上运行2.1、查看 Amazon SAM template2.2、编译和部署到 Amazon Lambda2.3、功能测试与验证 三、对比 Snapstart 效果四、资源清理五、实验总结总结 前言 在这个环节中,我们将延续《白鹿 Hands-on:消除冷启动——…...
ROC曲线
文章目录 前言一、ROC的应用?二、使用方式1. 数据准备2.绘图可视化 前言 在差异分析中,ROC曲线可以用来评估不同组之间的分类性能差异。差异分析旨在比较不同组之间的特征差异,例如在基因表达研究中比较不同基因在不同条件或组织中的表达水平…...
c++ 位图和布隆过滤器
位图(bitmap) 定义 位图是一种使用位数组存储数据的结构。每一位表示一个状态,通常用于快速判断某个值是否存在,或者用来表示布尔类型的集合。 特点 节省空间:一个字节可以表示8个状态。高效操作:位操作…...
阿里云CPU过载的一点思考
现象:阿里云ECS服务器连续5个周期CPU超90%告警 分析: max_connections和max_user_connections都做了限制,但是依然告警,服务器上有四个子服务,查看了每个服务的配置文件,发现使用同一个数据库账号&#x…...
单片机学习笔记 15. 串口通信(理论)
更多单片机学习笔记:单片机学习笔记 1. 点亮一个LED灯单片机学习笔记 2. LED灯闪烁单片机学习笔记 3. LED灯流水灯单片机学习笔记 4. 蜂鸣器滴~滴~滴~单片机学习笔记 5. 数码管静态显示单片机学习笔记 6. 数码管动态显示单片机学习笔记 7. 独立键盘单片机学习笔记 8…...
算法训练营day22(二叉树08:二叉搜索树的最近公共祖先,插入,删除)
第六章 二叉树part08 今日内容: ● 235. 二叉搜索树的最近公共祖先 ● 701.二叉搜索树中的插入操作 ● 450.删除二叉搜索树中的节点 详细布置 235. 二叉搜索树的最近公共祖先 相对于 二叉树的最近公共祖先 本题就简单一些了,因为 可以利用二叉搜索树的…...
Linux history 命令详解
简介 history 命令显示当前 shell 会话中以前执行过的命令列表。这对于无需重新输入命令即可重新调用或重新执行命令特别有用。 示例用法 显示命令历史列表 history# 示例输出如下:1 ls -l 2 cd /var/log 3 cat syslog执行历史记录中的命令 !<number>…...
Kafka知识体系
一、认识Kafka 1. kafka适用场景 消息系统:kafka不仅具备传统的系统解耦、流量削峰、缓冲、异步通信、可扩展性、可恢复性等功能,还有其他消息系统难以实现的消息顺序消费及消息回溯功能。 存储系统:kafka把消息持久化到磁盘上,…...
【Android】EventBus的使用及源码分析
文章目录 介绍优点基本用法线程模式POSTINGMAINMAIN_ORDEREDBACKGROUNDASYNC 黏性事件 源码注册getDefault()registerfindSubscriberMethods小结 postpostStickyunregister 介绍 优点 简化组件之间的通信 解耦事件发送者和接收者在 Activity、Fragment 和后台线程中表现良好避…...
【大数据学习 | Spark调优篇】Spark之内存调优
1. 内存的花费 1)每个Java对象,都有一个对象头,会占用16个字节,主要是包括了一些对象的元信息,比如指向它的类的指针。如果一个对象本身很小,比如就包括了一个int类型的field,那么它的对象头实…...
Linux:文件系统inode
早期,存储文件的设备是磁盘(当下的市场几乎都是SSD),但大家习惯的把它们都称为磁盘,磁盘是用来表示区分内存的存储设备。而在操作系统看来,这个存储设备的结构就是一个线性结构,这一点很重要。 …...
力扣难题解析
滑动窗口问题 76.最小覆盖子串 题目链接:76. 最小覆盖子串 - 力扣(LeetCode) 题目描述: 给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空…...
4.5-Channel 和 Flow:SharedFlow 和 StateFlow
文章目录 SharedFlow数据流的收集和事件订阅的区别launchIn() 和 shareIn() 的区别SharedFlow 与 Flow、Channel 的区别shareIn() 适用场景 shareIn() 的具体参数说明shareIn() 的 replay 参数shareIn() 的 started 参数WhileSubscribed() 的参数及适用场景 MutableSharedFlow、…...
Qt | TCP服务器实现QTcpServer,使用线程管理客户端套接字
点击上方"蓝字"关注我们 01、QTcpServer >>> QTcpServer 是 Qt 网络模块中的一个类,用于实现TCP服务器。它允许创建一个服务器,可以接受来自客户端的连接。QTcpServer 是事件驱动的,这意味着它将通过信号和槽机制处理网络事件。 常用函数 构造函数: QT…...
【提高篇】3.6 GPIO(六,寄存器介绍,下)
目录 2.3 输出速度寄存器OSPEEDR(GPIOx_OSPEEDR) (x = A..I) 2.4 上拉/下拉寄存器 (GPIOx_PUPDR) (x = A..I) 2.5 输入数据寄存器(IDR) 2.6 输出数据寄存器(ODR) 2.7 置位/复位寄存器(BSRR) 2.8 BSRR与ODR寄存器的区别 2.3 输出速度寄存器OSPEEDR(GPIOx_OSPEEDR) (…...
【AI】数据,算力,算法和应用(3)
三、算法 算法这个词,我们都不陌生。 从接触计算机,就知道有“算法”这样一个神秘的名词存在。象征着专业、权威、神秘、高难等等。 算法是一组有序的解决问题的规则和指令,用于解决特定问题的一系列步骤。算法可以被看作是解决问题的方法…...
深度学习笔记——生成对抗网络GAN
本文详细介绍早期生成式AI的代表性模型:生成对抗网络GAN。 文章目录 一、基本结构生成器判别器 二、损失函数判别器生成器交替优化目标函数 三、GAN 的训练过程训练流程概述训练流程步骤1. 初始化参数和超参数2. 定义损失函数3. 训练过程的迭代判别器训练步骤生成器…...
网络安全开源组件
本文只是针对开源项目进行收集,如果后期在工作中碰到其他开源项目将进行更新。欢迎大家在评论区留言,您在工作碰到的开源项目。 祝您工作顺利,鹏程万里! 一、FW(防火墙) 1.1 pfSense pfSense项目是一个免费…...
Python毕业设计选题:基于django+vue的智慧社区可视化平台的设计与实现+spider
开发语言:Python框架:djangoPython版本:python3.7.7数据库:mysql 5.7数据库工具:Navicat11开发软件:PyCharm 系统展示 管理员登录 管理员功能界面 养老机构管理 业主管理 社区安防管理 社区设施管理 车位…...
Oracle LinuxR7安装Oracle 12.2 RAC集群实施(DNS解析)
oracleLinuxR7-U6系统Oracle 12.2 RAC集群实施(DNS服务器) 环境 RAC1RAC2DNS服务器操作系统Oracle LinuxR7Oracle LinuxR7windows server 2008R2IP地址172.30.21.101172.30.21.102172.30.21.112主机名称hefei1hefei2hefei数据库名hefeidbhefeidb实例名…...
M2芯片安装es的步骤
背景:因为最近经常用到es,但是测试环境没有es,自己本地也没安装,为了方便测试,然后安装一下,但是刚开始安装就报错,记录一下,安装的版本为8.16.1 第一步:去官网下载maco…...
macos下brew安装redis
首先确保已安装brew,接下来搜索资源,在终端输入如下命令: brew search redis 演示如下: 如上看到有redis资源,下面进行安装,执行下面的命令: brew install redis 演示效果如下: …...
第六届金盾信安杯-SSRF
操作内容: 进入环境 可以查询网站信息 查询环境url https://114.55.67.167:52263/flag.php 返回 flag 就在这 https://114.55.67.167:52263/flag.php 把这个转换成短连接,然后再提交 得出 flag...
十大采购平台/seo代理
MyBatis中mapper.xml中foreach的使用 Author:kak MySql的动态语句foreach,当传入参数为数组或者集合时需要通过foreach标签进行遍历,其主要是在in条件中,可以在SQL语句中迭代一个集合; 综述 <foreach collection&…...
长宁区科技网站建设/兰州网站seo优化
什么是迭代器模式所谓迭代器模式,就是提供一种方法顺序访问一个容器对象的各个元素,而又不需要暴露该对象的内部表示的设计模式。迭代器模式的实现方式在客户访问类与容器体之间插入一个第三者——迭代器,就可以在不暴露该对象的内部表示的同…...
榆林做网站多少钱/网站免费推广网站
black_box_pad_pin声明用户定义的黑盒的管脚,作为外部环境可见的I/O pad,如果有不止一个端口,列在双引号内,以逗号分开。一般不需要这一属性,Synplify提供了预定义的I/Os。其语法如下object /* synthesis syn_black_bo…...
东莞商业网站建设常识/下载app
Configuration cfg new Configuration().configure() ;SchemaExport export new SchemaExport(cfg);export.create(true, true);...
wordpress微博登录注册/杭州seo网站推广排名
Python:3.7.1 Jupyter Notebook:6.3.0 Jupyter Notebook中将ipynb文件转为py文件前言一、将.ipynb文件转为.py文件1.命令行2.Jupyter Notebook页面上二、将.py文件转为.ipynb文件 (Jupyter Notebook加载.py文件)前言 Jupyter Notebook中属于网络页面交互…...
昆山靠谱的网站建设公司/为什么中国禁止谷歌浏览器
一、简介Visual Studio 2005 Team System(VSTS)能够为一个典型的软件开发团队中的每一种角色(架构师,工程经理,开发者和测试者)提供相应的工具。例如,它支持工程经理使用他们所熟知的工具&#…...