单例模式->饿汉模式->懒汉模式->阻塞队列->模拟实现阻塞队列->生产者消费者模型

单例模式->是一种固定套路,类似于"棋谱",按照套路来,可以避免一些问题

单例模式的特点->能够保证在某个类中只存在一个实例,不会创建多个实例

饿汉模式(线程安全):最基础的单例模式,类加载的同时就会创建实例,是线程安全的

public class Singleton {// 在类加载时就完成了实例化,避免了线程同步的问题private static Singleton instance = new Singleton();private Singleton() {}// 私有构造函数,防止被外部实例化public static Singleton getInstance() {// 获取单例对象的静态方法return instance;}
}

懒汉模式 (线程不安全):类加载的时候不会创建实例,第一次使用的时候才创建实例,线程不安全

public class SingleLaze {private static volatile SingleLaze instance = null;private SingleLaze() {} // 私有构造函数,防止外部实例化public static SingleLaze getInstance() {if (instance == null) { //首次调用get方法instance==null才会创建实例instance = new SingleLaze();           }return instance;}
}

懒汉模式(线程安全版)

两个懒汉模式横向对比可以发现,修改后的加上了锁和双重if判定以及给instance加上lvolatile 

  • 加锁/解锁在懒汉模式只会发生在第一次创建实例的时候,后面使用的时候就不需要加锁
  • 外层的if是判断当前是否已有实例对象
  • 内层if是判定是否需要创建对象,由于第一个if语句和第二个if语句之间 因为synchronized发生的阻塞过程中,期间可能instance被其他线程创建了实例,所以双重判定在多线程/阻塞中很有必要
  • volatile修饰instance确保多线程情况下的内存可见性和禁止指令重排序
public class SingleLaze {private static volatile SingleLaze instance = null;private SingleLaze() {} // 私有构造函数,防止外部实例化public static SingleLaze getInstance() {if (instance == null) { // 第一次检查,避免已有实例对象,再次创建新的对象情况synchronized (SingleLaze.class) { // 使用类对象作为锁,里面表示了singlelaze这个类if (instance == null) { // 第二次检查,确保单例instance = new SingleLaze();}}}return instance;}
}

阻塞队列是什么?

  • 阻塞队列是一种线程安全的数据结构,并具有以下特性
  • 当队列空的时候,出队列操作就会阻塞,直到有元素入队列
  • 当队列满的时候,入队列操作就会阻塞,直到有元素出队列

为什么要用阻塞队列?

在包饺子的场景中,如果不用阻塞队列,擀饺子皮的一直擀,但是包饺子的包的很慢,会出现饺子皮摆不下桌子的情况,如果用到阻塞队列,擀的人快,在桌子摆不下的情况就会阻塞等待,直到桌子能摆的下再擀,而包饺子快的情况下,就会等有饺子皮再包

标准库中阻塞队列BlockingQueue

BlockingQueue<String> queue=new ArrayBlockingQueue<String>(1000);
//创建新的阻塞队列,元素类型string,元素个数1000queue.put("abc");//put阻塞式入队列
// 需要throws声明可能会抛出异常String elem=queue.take();//阻塞式出队列
//elem元素

 模拟实现阻塞队列

  1. 使用循环队列的方式来实现
  2. put插入元素的时候,判定如果队列满了,就进行wait(需要注意要循环进行wait,可能在被唤醒的时候队列依旧是满的)
  3. take取出元素的时候,判定如果队列为空,就进行wait(也需要循环wait,确保线程在唤醒后能够再次检查等待条件,从而避免虚假唤醒等问题)
  4.  唤醒需要注意逻辑问题,wait是在满了或者空了的情况下才会运行,并且数据不可能是又满又空的状态,所以一次只会唤醒一个wait,两个wait不会同时运行,只有put下面的notify才能唤醒take上的wait,同理也只有take下面的notify才能唤醒put的wait
  5. 注意while和读写操作都要上锁,并且是一把锁才有用

代码实现

public class MyBlockingQueue1 {//为了简单,不写作泛型,只考虑元素类型是stringprivate String[]elems=null;private int head=0;//指向头结点下标private int tail=0;//指向尾节点下标private int size=0;//当前数组所存的元素个数//准备锁对象private Object locker =new Object();public MyBlockingQueue1(int capacity){//使用构造函数指定阻塞队列的最大容量elems=new String[capacity];}public void put(String elem) throws InterruptedException {synchronized (locker){while (size>=elems.length){locker.wait();//当前所占容量等于或超过最大容量,就阻塞等待}//没满elems[tail++]=elem;//将传入的元素从尾节点插入到数组,同时尾节点++if(tail>= elems.length){tail=0;}size++;//元素个数++//入队成功后唤醒可能在take时候阻塞的waitlocker.notify();}}public String take() throws InterruptedException {String elem=null;//在锁里面创建对象就在锁外面返回不了synchronized (locker){while (size==0){locker.wait();//当前数组元素为0,就阻塞等待}//没空elem=elems[head++];if(head>= elems.length){head=0;}size--;//元素个数++//入队成功后唤醒可能在take时候阻塞的waitlocker.notify();}return elem;}public static void main(String[] args) throws InterruptedException {MyBlockingQueue1 queue=new MyBlockingQueue1(1000);queue.put("aaa");queue.put("bbb");queue.put("ccc");queue.put("ddd");String elem= queue.take();System.out.println("elem="+elem);elem= queue.take();System.out.println("elem="+elem);elem= queue.take();System.out.println("elem="+elem);elem= queue.take();System.out.println("elem="+elem);}
}

运行结果

使用模拟实现的阻塞队列实现生产者消费者模型

 只需要更改主函数内容

public static void main(String[] args)  {MyBlockingQueue1 queue=new MyBlockingQueue1(1000);//生产者Thread t1=new Thread(()->{int n=1;while (true){try {queue.put(n+"");Thread.sleep(500);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("生产元素"+n);n++;}});//消费者Thread t2=new Thread(()->{while (true){try {String n= queue.take();System.out.println("消费元素"+n);//    Thread.sleep(500);} catch (InterruptedException e) {e.printStackTrace();}}});t1.start();t2.start();}
}

如果在生产后加上延时,消费后不加延时,就会出现生产一个消费一个的情况

如果在消费后面加上延时而生产后面不加延时,就会出现生产了1000多个才开始消费第一个的情况

 

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

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

相关文章

Flutter应用开发:掌握StatefulWidget的实用技巧

前言 随着移动应用的日益复杂&#xff0c;状态管理成为了 Flutter 应用开发中的一项重要挑战。 状态&#xff0c;即应用中的可变数据&#xff0c;它驱动着用户界面的渲染和交互。 在 Flutter 这样的声明式 UI 框架中&#xff0c;如何高效、可维护地管理状态&#xff0c;对于…

cuda中的cooperative_groups

背景 最近看到一个代码cooperative_groups.this_grid().sync()很好奇&#xff0c;这里好好梳理一下 分析 以前block内部的同步是用syncthreads(), block之间没有提供同步的接口&#xff0c;这样是合理的&#xff0c;假如有block间同步API的话&#xff0c;如果block太多&…

vue程序中如何设置调用springboot服务的url

在Vue程序中调用Spring Boot服务的URL&#xff0c;可以通过以下步骤实现&#xff1a; 安装Axios: Axios是一个基于Promise的HTTP库&#xff0c;可以用于浏览器和Node.js。可以使用npm或yarn安装Axios。 npm install axios # or yarn add axios创建Axios实例: 为了方便管理和复用…

Python--循环控制语句:continue 和 break

在Python编程中&#xff0c;continue 和 break 是两个非常有用的循环控制语句&#xff0c;它们允许我们以不同的方式控制循环的执行流程。 continue 语句 continue 用于中断当前循环的剩余部分&#xff0c;直接进入下一次循环的开始。它的作用是跳过当前循环中剩余的代码&…

GuLi商城-商品服务-API-属性分组-分组修改级联选择器回显

前端代码:略 后端回显接口: 递归方法: @Override publi

算法模板之单调栈【java】

算法模板之单调栈【java】 单调栈&#xff1a;在一维数组中找第一个满足某种条件的数找到数组中每个数【左侧】第一个【大于】它的数找到数组中每个数【左侧】第一个【大于或等于】它的数找到数组中每个数【左侧】第一个【小于】它的数找到数组中每个数【左侧】第一个【小于或等…

C# 反射详解

本文主要是对反射进行详细介绍&#xff0c;具体可以参照微软官方文档 首先我们来定义一个类型 public class Calculator {private int _number1 10;private int _number2 20;public int Number1 { get > _number1; set > _number1 value; }public int Number2 { get…

jquery中pdf在页面的显示和导出

jquery中pdf在页面的显示和导出 01 显示pdf01 .pdf结尾在线接口显示到页面 &#xff08;pdf.js库怎么安装及使用&#xff09;&#xff1a;只显示一页02 如何用PDF.JS显示整个PDF (而不仅仅是一页)&#xff1f;03 jQuery实现在线预览PDF文件(通过a标签链接跳转)&#xff1a; 02 …

RocketMq源码解析十二:消息消费负载

RocketMQ消息队列重新分配是由RebalancService线程来实现,一个MQClientinstance持有一个RebalanceService的实现,并随着MQClientInstance的启动而启动。我们看下下面的代码 位置:MQClientInstance:start方法 public void start() throws MQClientException {synchronized (…

‍我想我大抵是疯了,我喜欢上了写单元测试

前言 大家好我是聪。相信有不少的小伙伴喜欢写代码&#xff0c;但是对于单元测试这些反而觉得多此一举&#xff0c;想着我都在接口文档测过了&#xff01;还要写什么单元测试&#xff01;写不了一点&#xff01;&#xff01; 由于本人也是一个小小程序猿&#x1f649;&#xf…

关于HDFS、Hive和Iceberg

HDFS & Hive 如果我们将Hive比喻为储藏室&#xff0c;那么HDFS&#xff08;Hadoop Distributed File System&#xff09;就可以比作是储藏室所在的建筑物的地基和结构。 HDFS是一个分布式文件系统&#xff0c;它的设计目标是存储和管理海量数据。在我们的类比中&#xff…

华为云SQLServer 慢日志查看

作者&#xff1a;梦莱 1、背景 华为云目前只支持 SQLServer 登录数据库&#xff0c;不支持查看慢日志。对于开启慢日志的实例&#xff0c;也只能通过将慢日志下载到本地 再远程连接目标实例数据库查看。本篇将华为云 SQLServer 实例出现资源异常&#xff0c;排查问题的方案整…

QT5_C++基础

1. 什么是类和对象 C的类是一种构造类型&#xff0c;与C语言的结构体类似&#xff0c;但是进行了一些拓展&#xff0c;类的成员不但可以是变量&#xff0c;还可以是函数&#xff1b;通过类定义出来的变量也有特定的称呼&#xff0c;叫做“对象”类是创建对象的模板&#xff0c…

【java】力扣 合法分割的最小下标

文章目录 题目链接题目描述思路代码 题目链接 2780.合法分割的最小下标 题目描述 思路 这道题是摩尔算法的一种扩展 我们先可以找到候选人出来&#xff0c;然后去计算他在左右两边元素出现的次数&#xff0c;只有当他左边时&#xff0c;左边出现的次数2 >左边的长度&…

【 LCD1602显示屏】使用STC89C51控制1602显示、读写操作时序

文章目录 LCD1602显示概述&#xff1a;引脚说明控制指令接线 控制思路步骤 代码示例总结对databuffer dataShow;的理解 LCD1602显示 概述&#xff1a; LCD1602&#xff08;Liquid Crystal Display&#xff09;是一种工业字符型液晶&#xff0c;能够同时显示 1602 即 32 字符…

Android Studio的xml文件的layout布局,在添加属性的过程中,没有自动补全代码问题的解决方案

在build.gradle文件中&#xff0c;把compileSdkVersion和targetSdkVersion两个参数改成32就好了。 参考&#xff1a;关于Android Studio的xml文件的layout布局&#xff0c;在添加属性的过程中&#xff0c;不显示提示词&#xff08;没有自动补全代码&#xff09;的问题的解决方…

SpringBoot增加网关服务

一、新建gateway项目 二、添加依赖 dependencies {implementation org.springframework.cloud:spring-cloud-starter-gateway:4.0.0 } 三、增加路由规则配置 一个web服务、一个service服务 bootstrap.yaml&#xff1a; server:port: 80 spring:application:name: gatewayc…

子树的重心

描述 输入一棵树,判断每一棵子树的重心是哪一个节点。 输入描述 第一行输入n,q。n表示树的节点个数&#xff0c;q表示询问次数 第二行n-1个数&#xff0c;分别表示从节点2开始&#xff0c;各节点的父亲节点。 后面q行&#xff0c;每行一个数x&#xff0c;表示询问当前以x为根…

【STM32 HAL库】I2S的使用

使用CubeIDE实现I2S发数据 1、配置I2S 我们的有效数据是32位的&#xff0c;使用飞利浦格式。 2、配置DMA **这里需要注意&#xff1a;**i2s的DR寄存器是16位的&#xff0c;如果需要发送32位的数据&#xff0c;是需要写两次DR寄存器的&#xff0c;所以DMA的外设数据宽度设置16…

入门C语言只需一个星期(星期二)

点击上方"蓝字"关注我们 01、算术运算符 int myNum = 100 + 50;int sum1 = 100 + 50; // 150 (100 + 50)int sum2 = sum1 + 250; // 400 (150 + 250)int sum3 = sum2 + sum2; // 800 (400 + 400) + 加 将两个值相加 x + y - 减 从另一个值中减去一个值 …