JavaEE——No.1 多线程案例

news/2024/4/30 3:17:56/文章来源:https://blog.csdn.net/m0_58592142/article/details/126999401
JavaEE传送门

JavaEE

JavaEE——No.1 线程安全问题

JavaEE——No.2 线程安全问题


目录

  • 多线程案例
    • 1. 单例模式
      • 饿汉模式
      • 懒汉模式
    • 2. 阻塞队列
      • 阻塞队列的使用
      • 阻塞队列的实现


多线程案例

1. 单例模式

单例模式是一种常见的设计模式.

设计模式: 软件开发时, 会遇到一些常见的 “问题场景”, 针对这些问题, 就有大佬们整理出来了一些相对应的解决办法, 供其他人来研究学习.

单例

指单个实例, 某个类, 有且只有一个实例. 有些场景, 要求实例不能有多个, 通过单例模式, 相当于对 “单个实例” 做了一个更加严格的约束.

比如JDBC中的 DataSource 实例就只需要一个.

单例模式具体的实现方式, 又分为 “饿汉模式” 和 “懒汉模式” 两种.

static (Java 中天然的单个实例)

static 修饰的成员/属性, 变成了类成员/类属性. 当属性变成类属性的时候, 此时就是一个 “单个实例”

class Singleton {private static Singleton instance = new Singleton();
}

上述代码中的 instance 就是 Signleton 这个类的唯一实例.


饿汉模式

类加载的同时, 创建实例.

class Singleton {private static Singleton instance = new Singleton();public Singleton getInstance() {return instance;}//把构造方法设为私有的, 此时在类的外面, 就无法 new 实例了private Singleton() {//强制禁止其他的类中, new Singleton 实例//唯一的入口就是通过 getInstance 方法}
}

懒汉模式

类加载的时候不创建实例. 第一次使用的时候才创建实例

(创建实例的实际更迟, 效率更高)

#单线程版

class SingletonLazy {private static SingletonLazy instance = null;public static SingletonLazy getInstance() {if (instance == null) {//首次调用instance的时候才会触发instance = new SingletonLazy();}return instance;}private SingletonLazy() {}
}public class Test {public static void main(String[] args) {SingletonLazy instance = SingletonLazy.getInstance();//如果在整个代码中后续没人调用 getInstance, 这样就节省了构造实例的过程}
}

# 注意 #

在上述代码中, getInstance 方法中, 及涉及到了读, 也涉及到了写, 是不安全的.

如下图 t2 线程进行 LOAD 的时候, t1 还没有进行修改完, t2 读到的就还是之前的值, 就会导致同时创建两个实例.

#多线程版

我们这时使用 synchronized 来改善这里的线程安全问题

class SingletonLazy {private static SingletonLazy instance = null;public static SingletonLazy getInstance() {   synchronized (SingletonLazy.class) { if (instance == null) {instance = new SingletonLazy();}}return instance;}private SingletonLazy() {}
}

但这时, 我们只要调用 getInstance 方法, 就会进行加锁, 加锁绝对不是无脑加的. 加锁的开销还是比较大的.

实例创建之前, 是线程不安全的, 需要加锁. 实例创建之后, 是线程安全的, 就不需要加锁了.

#多线程改进版

因此, 我们可以在加锁的外层, 再加上一层判定.

class SingletonLazy {private static SingletonLazy instance = null;public static SingletonLazy getInstance() {// 判定是否要加锁if (instance == null) {synchronized (SingletonLazy.class) {// 判定是否要创建实例if (instance == null) {instance = new SingletonLazy();}}}return instance;}private SingletonLazy() {}
}

但这时可能还会出现指令重排序的问题, 导致线程不安全

new 操作的本质分为三个步骤

  1. 申请内存, 得到内存首地址
  2. 调用构造方法来初始化实例
  3. 把内存的首地址赋值给 instance 引用

比如我们在进行

instance = new SingletonLazy();

操作时, 在单线程的角度下, 2 和 3 是可以调换顺序的. (在单线程的情况下, 此时 2 和 3 先执行谁, 后执行谁效果是一样的)

假设此处触发了指令重排序, 有可能 t1 执行了 1 和 3 之后 (得到了不完全的对象, 只是有内存, 内存上的数据无效) , 执行 2 之前

这时 t2 线程调用了这个 getlnstance ,这个 getlnstance 就会认为这个 Instance 非空, 就直接返回了.

并且在后续可能会针对 Instance 进行解引用操作 (使用里面的属性/ 方法).

#多线程最终版

我们想要禁止指令重排序, 需要使用 volatile 关键字

class SingletonLazy {//禁止指令重排序private volatile static SingletonLazy instance = null;public static SingletonLazy getInstance() {if (instance == null) {synchronized (SingletonLazy.class) {if (instance == null) {instance = new SingletonLazy();}}}return instance;}private SingletonLazy() {}
}

2. 阻塞队列

  1. 阻塞队列是一个特殊的队列, 遵循先进先出的原则
  2. 是线程安全的
  3. 如果队列满, 继续入队列, 入队列操作就会阻塞. 直到队列不满, 入列才能完成
  4. 如果队列空, 继续出队列, 出队列操作就会阻塞, 直到队列不空, 出队列才能完成.

阻塞队列的应用场景: 生产者消费者模型

生产者消费者模式就是通过一个容器来解决生产者和消费者的强耦合问题。

生产者和消费者彼此之间不直接通讯,而通过阻塞队列来进行通讯,所以生产者生产完数据之后不用等待消费者处理,直接扔给阻塞队列,消费者不找生产者要数据,而是直接从阻塞队列里取.

1) 阻塞队列使生产者和消费者之间解耦合, 生产者和消费者就不需要知道对方的存在.

2) 削峰填谷 (按照没有生产者消费者模型的写法, 外面流量过来的压力, 就会直接压到每个服务器上, 如果某个服务器抗压能力不行, 就容易挂)

此时的阻塞队列, 就像是三峡的水坝, 如果上游水大, 就关闸蓄水. 如果上游水少, 就开闸放水.


阻塞队列的使用

Java 标准库中内置了阻塞队列.

  • BlockingQueue 是一个接口

队列有三个基本操作: 入队列, 出队列, 取队首元素 (阻塞队列没有专门的带阻塞的取队首元素)

  • 入队列 .put( )

(阻塞队列也有入队列也可以使用 offer 方法, 但并不推荐使用, 因为其不带有阻塞特性)

// 带阻塞功能
blockingQueue.put(1);
blockingQueue.put(2);
blockingQueue.put(3);
  • 出队列 .take( )
Integer ret = blockingQueue.take();
System.out.println(ret);
ret = blockingQueue.take();
System.out.println(ret);
ret = blockingQueue.take();
System.out.println(ret);
  • 阻塞

当我们在上述操作中, 已经把1, 2, 3 全部取出的时候, 在进程 take( ) 操作就会进行阻塞.


阻塞队列的实现

  • 通过数组实现
  • 入队列和出队列方法中都涉及到修改, 防止出现多线程安全问题, 使用 synchronized 进行加锁
  • put 插入元素时, 如果队列满了, 进行 wait
  • take 取出元素时, 如果队列为空, 进行 wait
  • 一个队列, 不会出现又是空又是满的情况, 所以不需要两个锁对象
  • wait 外面需要加上 while 循环, 在多线程的情况下, 被唤醒时不一定队列就满或者不满
class MyBlockingQueue {private int[] items = new int[1000];private volatile int head = 0;private volatile int tail = 0;private volatile int size = 0;//入队列public void put(int elem) throws InterruptedException {synchronized (this) {//队列满了, 进行 waitwhile (size >= items.length) {this.wait();}items[tail] = elem;tail++;if(tail >= items.length) {tail = 0;}size++;this.notify();}}// 出队列public Integer take() throws InterruptedException {synchronized (this) {//队列为空, 进行 waitwhile (size == 0) {this.wait();}int ret = items[head];head++;if(head >= items.length) {head = 0;}size--;this.notify();return ret;}}
}

🌷(( ◞•̀д•́)◞⚔◟(•̀д•́◟ ))🌷

以上就是今天要讲的内容了,希望对大家有所帮助,如果有问题欢迎评论指出,会积极改正!!
在这里插入图片描述
加粗样式

这里是Gujiu吖!!感谢你看到这里🌬
祝今天的你也
开心满怀,笑容常在。

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

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

相关文章

存储系统基本概念

内容框图 存储器的层次化结构 由于cpu运行太快,所以中间需要主存,Cache,寄存器去传递。 主存–辅存(硬件操作系统):实现虚拟存贮系统,解决了主存容量不够的问题。 Cache–主存(硬件自…

测试用例设计专栏

哈喽大家好哎呀,今天给大家普及一下测试用例如何设计,看牛逼的大佬们是如何测试的。 测试用例笔试题 出题:在一个页面上有一个输入框,一个计数器(count)按钮,用于计算一个文本字符串中字母a出现的次数,请…

Linux 逻辑卷精简卷报错问题解决

一、 故障描述 现象1:oraclelog目录提示坏道信息,进行修复后执行删除文件操作,目录不可使用。 现象2:lsblk看到目录出现重复,并且有tmeta,tdata卷出现(图一) 现象3:message日志出现多目录报错,持续写入(图二) 图一 检查lv #lvs -a 看到多出的pmspare,tdata,tmeta…

VSCode 使用教程-9.Node运行js出现 Cannot use import statement outside a module的问题

前言 js中导入公共模块,使用import的方式导入,用node运行js文件会出现Cannot use import statement outside a module的问题 问题描述 目录结构 └─src└─js└─ext.js└─main.js └─index.html在ext.js 文件写一些公共方法 export const m (f…

vue3 ts vite 项目快速构建

1.安装nodejs(建议装14版本稳定) 下载 | Node.js 中文网 装完之后会有一个命令叫 npm 可以在终端输入npm -v 来检查是否安装成功2.构建vite项目官方文档开始 {#getting-started} | Vite中文网 vite 的优势冷服务 默认的构建目标浏览器是能 在 script 标签上支持原生 ESM 和…

Java操作HDFS

1. 创建maven项目 New Project 2. 添加依赖 <!-- https://mvnrepository.com/artifact/org.apache.hadoop/hadoop-client --><dependency><groupId>org.apache.hadoop</groupId><artifactId>hadoop-client</artifactId><version>2.…

steam搬砖汇率差项目详解

很久没有分享赚钱项目了&#xff0c;今天这里给大家介绍一个游戏搬砖的项目&#xff1a;steam游戏汇率差赚钱项目。 项目原理&#xff1a; Steam平台是一个国外游戏及装备售卖平台。而我们所谓的Steam游戏装备搬砖就是利用steam平台和网易的BUFF平台来操作。steam汇率差赚钱原…

十大经典排序算法综述(Java代码实现,思想通用)

关于十大排序的文章也有不少了&#xff0c;但感觉大部分在各个排序算法的适用场景、如何实现外排等细节方面没怎么讲&#xff0c;故总结了这篇文章&#xff0c;欢迎浏览 一、前言 内部排序是指排序时将待排序数据全部加载到内存的算法。 外部排序是指在处理海量数据排序时&…

什么是C语言?

什么是C语言&#xff1f; 文章目录什么是C语言&#xff1f;1.C语言的起源2.C语言的使用领域3. 为什么要学习C语言4.C语言的学习境界5.如何学习C语言6.学习C语言的推荐书籍1.C语言的起源 C语言之父是丹尼斯里奇&#xff1a;丹尼斯里奇&#xff08;1941年9月9日-2011年10月12日&…

Linux 简单命令 - cron 计划任务 、NTP

Linux 简单命令 - cron NTP cronNTP 一、cron 计划任务就是按照系统的时间(时刻、周期)执行指定的任务 系统服务&#xff1a; crond。 配置文件&#xff1a; /etc/crontab /var/spool/cron/用户名 配置记录格式&#xff1a; 分 时 日 月 周 任务操作命令 (用绝对路径、必要时…

集成学习详解

入门小菜鸟&#xff0c;希望像做笔记记录自己学的东西&#xff0c;也希望能帮助到同样入门的人&#xff0c;更希望大佬们帮忙纠错啦~侵权立删。 目录 一、集成学习的产生原因与相关定义 1、产生原因 2、相关定义 &#xff08;1&#xff09;同质集成 &#xff08;2&#xf…

第三章 学校与班级管理

01 学校组织与管理 02 班级与班集体 03 班主任与班主任工作 04 班级活动与班队活动 05 课外活动 02 班级与班集体 一、班级与班集体 二、班级管理 三、班级突发事件的处理 一、班级与班集体 &#xff08;一&#xff09;班级 了解 年龄、知识程度相近&#xff0c;有共同的学…

python学习—第一步—Python小白逆袭大神(第二天)

python进阶python语法继续学习数据结构数字字符串列表元组字典面向对象继承JSON异常处理try except finallyLinux命令作业来啦&#xff01;问题python语法继续学习 数据结构 数字 Number类型用于存储数值。 1、数学运算math模块及常用函数 菜鸟教程 导入math 代码示例&#…

微信小程序事件和页面跳转

一、页面跳转 1.非TabBar页面 一个小程序拥有多个页面&#xff0c;我们通过wx.navigateTo进入一个新的页面 我们通过下边点击事件实现页面跳转进行代码实现及参考 wx.navigateBack&#xff08;&#xff09;回退到上一个页面 wx.redirectTo&#xff08;url&#xff09;删除…

最新AWVS14.9.220913107 支持Windows使用教程(附下载地址)

主要内容&#xff1a;awvs14.9下载、awvs14.9使用教程、awvs14.9安装教程、awvs14.9批量扫描、awvs14.9用法、awvs最新版下载 使用方法 一键PJ 一键PJ则只需要在安装Acunetix软件后&#xff0c;运行pj工具即可 通用思路 修改hosts文件&#xff08;C:\Windows\System32\driv…

15天深度复习JavaWeb的详细笔记(十一)——VUE、Element

文章目录demo11-VUE、Element1&#xff0c;VUE1.1 概述1.2 快速入门1.3 Vue 指令1.3.1 v-bind & v-model 指令1.3.2 v-on 指令1.3.3 条件判断指令1.3.4 v-for 指令1.4 生命周期2&#xff0c;Element2.1 快速入门2.2 Element 布局2.2.1 Layout 局部2.2.2 Container 布局容器…

Spring 注解

事务注解 使用注解 EnableTransactionManagement 开启事务支持后&#xff0c;然后在访问数据库的Service方法上添加注解 Transactional 便可 * EnableTransactionManagement&#xff0c;会额外加载 4 个 bean * BeanFactoryTransactionAttributeSourceAdvisor 事务切面类 …

C# ASP.NET Web Core API (.NET 6.0)

目录 一、简介 二、创建项目 三、启动项目 四、开放访问权限 五、添加其他的API 结束 一、简介 ASP.NET Core Web API 是 ASP.NET Core MVC 的一个功能。ASP.NET Core MVC 包含了对 Web API 的支持。可以构建多种客户端的 HTTP 服务。ASP.NET Core Web API可用于在 .NET…

LeetCode 0328. 奇偶链表

【LetMeFly】328.奇偶链表 力扣题目链接&#xff1a;https://leetcode.cn/problems/odd-even-linked-list/ 给定单链表的头节点 head &#xff0c;将所有索引为奇数的节点和索引为偶数的节点分别组合在一起&#xff0c;然后返回重新排序的列表。 第一个节点的索引被认为是 奇…

8-Arm PEG-ACA,8-Arm PEG-Acrylamide,八臂-聚乙二醇-丙烯酰胺可用于自由基引发剂

一&#xff1a;产品描述 1、名称 英文&#xff1a; 8-Arm PEG-ACA 8-Arm PEG-Acrylamide 中文&#xff1a; 八臂-聚乙二醇-ACA 2、CAS编号&#xff1a;N/A 3、所属分类&#xff1a;Acrylate/Acrylamide PEG Multi-arm PEGs 4、分子量&#xff1a;可定制 5、质量控制&a…