当前位置: 首页 > news >正文

【设计模式】单例模式和生产者消费者模型

单例模式

单例模式是一种常见的设计模式,旨在确保一个类只有一个实例,并提供一个全局访问点。这种模式常用于需要控制资源的类,比如配置管理、线程池等。

主要特性:

  • 唯一性:类只有一个实例。
  • 全局访问:提供一个静态方法获取该实例。
  • 延迟加载(可选):实例在第一次使用时创建。

实现方法

1. 饿汉式单例

在类加载时就创建实例,线程安全,但不支持延迟加载。

public class Singleton {// 静态实例private static final Singleton instance = new Singleton();// 私有构造函数private Singleton() {}// 公共方法获取实例public static Singleton getInstance() {return instance;}
}

2. 懒汉式单例

在第一次调用时创建实例,支持延迟加载,线程不安全的实现。

public class Singleton {// 静态实例private static Singleton instance;// 私有构造函数private Singleton() {}// 公共方法获取实例public static Singleton getInstance() {if (instance == null) {instance = new Singleton();}return instance;}
}

3. 线程安全的懒汉式单例

通过同步方法实现线程安全。

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

4. 双重检查锁定

结合懒加载和性能优化,通过“双重检查”实现线程安全,减少同步开销。

public class Singleton {private static volatile Singleton instance;private Singleton() {}public static Singleton getInstance() {if (instance == null) { // 第一次检查synchronized (Singleton.class) {if (instance == null) { // 第二次检查instance = new Singleton();}}}return instance;}
}

总结

单例模式通过限制实例化数量,简化了代码管理和资源控制。在选择实现方式时,要考虑到线程安全和性能需求,适当选择实现。

wait 和 notify 方法

在Java中,wait 和 notify 是用于线程间通信的重要方法。这些方法被定义在 Object 类中,因此所有的对象都可以利用这些方法进行线程协调。它们通常用于同步块(synchronized block)中,以实现线程的等待和通知机制。

wait 方法

wait() 方法使当前线程等待,直到其他线程调用 notify() 或 notifyAll() 方法来唤醒它。调用 wait() 方法的线程会释放持有的对象锁。

notify 方法

notify() 方法用于唤醒一个正在等待该对象监视器的线程。如果有多个线程在等待,则其中一个线程会被唤醒,具体被哪个线程唤醒是不确定的。使用 notifyAll() 可以唤醒所有在等待该对象监视器的线程。

生产者消费者模型

生产者消费者模型是一种常见的并发设计模式,它用于解决生产者和消费者之间的协作问题。生产者负责生成数据,消费者则负责处理数据。这个模型通过缓冲区的方式进行协调,确保生产者在缓冲区满时暂停生产,消费者在缓冲区空时暂停消费。

主要特性:

  1. 并发性:生产者和消费者可以并行工作。
  2. 缓冲区:使用一个共享的缓冲区来存储数据。
  3. 同步控制:需要通过适当的同步机制来确保线程安全。

实现方法

以下是一个使用 Java 的生产者消费者模型的示例,利用 wait() 和 notify() 方法进行线程间的通信。

1. 定义缓冲区

import java.util.LinkedList;class Buffer {private LinkedList<Integer> queue = new LinkedList<>();private final int CAPACITY = 5;// 生产者向缓冲区添加数据public synchronized void produce(int value) throws InterruptedException {while (queue.size() == CAPACITY) {wait(); // 如果缓冲区满,等待消费者消费}queue.add(value);System.out.println("Produced: " + value);notifyAll(); // 通知消费者可以消费}// 消费者从缓冲区获取数据public synchronized int consume() throws InterruptedException {while (queue.isEmpty()) {wait(); // 如果缓冲区空,等待生产者生产}int value = queue.removeFirst();System.out.println("Consumed: " + value);notifyAll(); // 通知生产者可以生产return value;}
}

2. 定义生产者

class Producer extends Thread {private Buffer buffer;public Producer(Buffer buffer) {this.buffer = buffer;}@Overridepublic void run() {for (int i = 0; i < 10; i++) {try {buffer.produce(i);Thread.sleep(500); // 模拟生产时间} catch (InterruptedException e) {e.printStackTrace();}}}
}

3. 定义消费者

class Consumer extends Thread {private Buffer buffer;public Consumer(Buffer buffer) {this.buffer = buffer;}@Overridepublic void run() {for (int i = 0; i < 10; i++) {try {buffer.consume();Thread.sleep(1000); // 模拟消费时间} catch (InterruptedException e) {e.printStackTrace();}}}
}

4. 测试生产者和消费者

public class ProducerConsumerExample {public static void main(String[] args) {Buffer buffer = new Buffer();Producer producer = new Producer(buffer);Consumer consumer = new Consumer(buffer);producer.start();consumer.start();}
}

总结

在这个示例中,生产者和消费者通过一个共享的缓冲区进行交互,使用 synchronized 关键字、wait() 和 notifyAll() 方法实现同步控制。生产者在缓冲区满时暂停生产,消费者在缓冲区空时暂停消费,从而有效地协调了两者之间的关系。这个模型广泛应用于多线程编程、任务调度等场景。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 使用Jupyter Notebook实现Python编程
  • mysql 悲观锁使用
  • 查找------折半查找(二分查找)
  • 掌握ThinkPHP6中的数据验证技巧,提升开发效率
  • 【PyTorch】深度学习PyTorch加载数据
  • 区块链国赛第六套样题(关于运维)
  • Java基础——自学习使用(多态)
  • TCP与UDP传输的学习
  • GraphQL:API开发的未来,重塑数据交互的艺术
  • 发条朋友圈赚900,这钱太好赚了吧?
  • 照片逼真肖像动画的音频驱动合成——AniPortrait翻译与调试
  • 【YOLO5 项目实战】(7)YOLO5 手势识别
  • 45+用户占比近30%,网文产业如何赋能IP长链?
  • 如何使用gewe开发微信机器人
  • 010 OSS文件上传
  • 【跃迁之路】【735天】程序员高效学习方法论探索系列(实验阶段492-2019.2.25)...
  • Angular 4.x 动态创建组件
  • CAP理论的例子讲解
  • Invalidate和postInvalidate的区别
  • java概述
  • Java新版本的开发已正式进入轨道,版本号18.3
  • Kibana配置logstash,报表一体化
  • k个最大的数及变种小结
  • leetcode386. Lexicographical Numbers
  • PHP 7 修改了什么呢 -- 2
  • Python语法速览与机器学习开发环境搭建
  • RxJS: 简单入门
  • 反思总结然后整装待发
  • 理解 C# 泛型接口中的协变与逆变(抗变)
  • 排序(1):冒泡排序
  • 前端 CSS : 5# 纯 CSS 实现24小时超市
  • 深入 Nginx 之配置篇
  • 线上 python http server profile 实践
  • 新手搭建网站的主要流程
  • 移动端 h5开发相关内容总结(三)
  • 在weex里面使用chart图表
  • 自动记录MySQL慢查询快照脚本
  • JavaScript 新语法详解:Class 的私有属性与私有方法 ...
  • PostgreSQL 快速给指定表每个字段创建索引 - 1
  • ​人工智能之父图灵诞辰纪念日,一起来看最受读者欢迎的AI技术好书
  • # 深度解析 Socket 与 WebSocket:原理、区别与应用
  • #Linux(权限管理)
  • $con= MySQL有关填空题_2015年计算机二级考试《MySQL》提高练习题(10)
  • (10)STL算法之搜索(二) 二分查找
  • (11)工业界推荐系统-小红书推荐场景及内部实践【粗排三塔模型】
  • (173)FPGA约束:单周期时序分析或默认时序分析
  • (2)MFC+openGL单文档框架glFrame
  • (aiohttp-asyncio-FFmpeg-Docker-SRS)实现异步摄像头转码服务器
  • (定时器/计数器)中断系统(详解与使用)
  • (附源码)ssm智慧社区管理系统 毕业设计 101635
  • (七)c52学习之旅-中断
  • (算法)求1到1亿间的质数或素数
  • (原創) 如何動態建立二維陣列(多維陣列)? (.NET) (C#)
  • (转)清华学霸演讲稿:永远不要说你已经尽力了
  • (转)视频码率,帧率和分辨率的联系与区别