设计模式精讲 Day 1:单例模式(Singleton Pattern)

【设计模式精讲 Day 1】单例模式(Singleton Pattern)


文章内容

开篇

在软件开发中,设计模式是解决常见问题的通用解决方案。作为“设计模式精讲”系列的第一天,我们将深入讲解单例模式(Singleton Pattern),这是创建型模式中最基础、最常用的一种。

单例模式的核心思想是:确保一个类只有一个实例,并提供一个全局访问点。它广泛应用于需要共享资源、控制资源访问或保证唯一性的场景,如数据库连接池、配置管理器、日志系统等。

本文将从理论到实践,全面解析单例模式的设计原理、实现方式、应用场景以及在Java中的实际应用,帮助你掌握这一经典设计模式,并在项目中灵活运用。


模式定义

单例模式是一种创建型设计模式,它确保一个类在整个应用程序中只存在一个实例,并提供一个全局访问该实例的方法。

其核心思想是:

  • 限制类的实例化次数,仅允许创建一次。
  • 提供一个全局访问点,方便其他对象获取该实例。

模式结构

单例模式的UML类图包含以下关键角色:

角色说明
Singleton单例类,负责控制实例的创建和访问

在代码中,Singleton 类通常包含以下元素:

  • 一个私有构造函数,防止外部直接实例化。
  • 一个静态的私有实例变量,用于保存唯一的实例。
  • 一个公共的静态方法(如 getInstance()),用于返回该实例。

适用场景

单例模式适用于以下典型场景:

场景描述
全局配置管理如应用的配置信息,只需加载一次
数据库连接池确保多个组件共享同一个数据库连接池
日志记录器保证所有模块使用同一个日志输出
缓存管理控制缓存数据的唯一性
资源管理器如线程池、网络连接等资源的统一管理

实现方式

下面是一个完整的Java实现示例,展示了单例模式的多种实现方式:

饿汉式单例(线程安全)

/*** 饿汉式单例:类加载时就初始化实例,线程安全*/
public class Singleton {// 私有静态实例,类加载时就初始化private static final Singleton instance = new Singleton();// 私有构造函数,防止外部实例化private Singleton() {}// 公共静态方法,返回唯一实例public static Singleton getInstance() {return instance;}// 示例方法public void showMessage() {System.out.println("This is a singleton instance.");}
}

懒汉式单例(非线程安全)

/*** 懒汉式单例:延迟初始化,但不保证线程安全*/
public class LazySingleton {private static LazySingleton instance;private LazySingleton() {}public static LazySingleton getInstance() {if (instance == null) {instance = new LazySingleton();}return instance;}public void showMessage() {System.out.println("Lazy singleton instance.");}
}

双重检查锁(DCL)单例(线程安全)

/*** DCL(双重检查锁)单例:线程安全且延迟初始化*/
public class DCLSingleton {private static volatile DCLSingleton instance;private DCLSingleton() {}public static DCLSingleton getInstance() {if (instance == null) {  // 第一次检查synchronized (DCLSingleton.class) {if (instance == null) {  // 第二次检查instance = new DCLSingleton();}}}return instance;}public void showMessage() {System.out.println("DCL singleton instance.");}
}

枚举单例(推荐方式)

/*** 枚举单例:线程安全、防止反射攻击、序列化安全*/
public enum EnumSingleton {INSTANCE;public void showMessage() {System.out.println("Enum singleton instance.");}
}

工作原理

单例模式通过以下机制实现:

  1. 限制构造函数访问:通过将构造函数设为私有,防止外部通过 new 创建实例。
  2. 静态实例变量:使用静态变量存储唯一实例,确保整个应用中只有一个实例。
  3. 静态访问方法:通过静态方法 getInstance() 提供对实例的访问,避免直接暴露构造函数。

在多线程环境下,需特别注意线程安全问题。例如,懒汉式单例未加锁会导致多个线程同时创建实例,而使用 volatile 和双重检查锁可以有效避免此问题。


优缺点分析

优点缺点
确保全局唯一性,便于资源共享过度使用可能导致耦合,难以测试
提高性能,减少资源开销不适合需要频繁创建和销毁的对象
易于维护和扩展不适合需要动态实例化的场景

案例分析:日志系统

在实际项目中,日志系统常使用单例模式来确保所有模块都使用同一个日志实例。

问题描述

在一个大型系统中,多个模块都需要记录日志,如果每个模块都独立创建日志对象,会导致资源浪费和日志信息不一致。

解决方案

使用单例模式创建一个全局的日志记录器,所有模块通过该实例进行日志操作。

/*** 日志记录器,使用单例模式*/
public class Logger {private static volatile Logger instance;private Logger() {}public static Logger getInstance() {if (instance == null) {synchronized (Logger.class) {if (instance == null) {instance = new Logger();}}}return instance;}public void log(String message) {System.out.println("[LOG] " + message);}
}// 使用示例
public class App {public static void main(String[] args) {Logger logger = Logger.getInstance();logger.log("Application started.");}
}

与其他模式的关系

单例模式与以下设计模式有密切关系:

模式关系说明
工厂模式工厂模式可以创建单例对象,但单例模式更关注唯一性
抽象工厂抽象工厂通常返回一组相关对象,而单例模式专注于单一对象
代理模式代理模式可以包装单例对象,增加额外功能
原型模式原型模式通过复制创建对象,而单例模式强调唯一性

总结

今天我们详细讲解了单例模式,包括它的定义、结构、适用场景、多种实现方式、工作原理、优缺点、真实案例以及与其他模式的关系。

通过本篇文章,你应该已经掌握了:

  • 单例模式的核心思想;
  • Java中不同实现方式的优劣;
  • 如何在实际项目中使用单例模式;
  • 单例模式在Java标准库和框架中的应用。

下一天,我们将进入“设计模式精讲”的第二天,讲解工厂方法模式(Factory Method Pattern),敬请期待!


标签

设计模式, 单例模式, Java, 设计模式精讲, 软件架构, 编程技术, 面向对象设计, Java设计模式


文章简述

本文是“设计模式精讲”系列的第一篇,深入讲解了单例模式(Singleton Pattern)。文章从理论出发,结合Java代码示例,详细阐述了单例模式的核心思想、实现方式、适用场景及优缺点。通过真实项目案例(如日志系统),展示了如何在实际开发中应用该模式。同时,文章还对比了单例模式与其他设计模式的关系,并探讨了其在Java标准库中的应用。本文旨在帮助开发者理解并掌握单例模式,提升代码质量和可维护性,为后续设计模式的学习打下坚实基础。


进一步学习资料

  1. 《设计模式:可复用面向对象软件的基础》 —— GoF 经典著作
  2. Java Design Patterns - Oracle Docs
  3. Design Patterns in Java - Baeldung
  4. Singleton Pattern - Refactoring Guru
  5. Java并发编程实战 - 《Java Concurrency in Practice》

核心设计思想总结

通过本文的学习,我们掌握了单例模式的核心思想:确保一个类只有一个实例,并提供全局访问点。该模式在资源管理、配置中心、日志系统等场景中具有重要价值。

在实际项目中,合理使用单例模式可以提高系统性能、降低资源消耗、增强代码可维护性。然而,也应注意避免滥用,特别是在需要频繁创建和销毁对象的场景中。

希望你在今后的开发中能够灵活运用单例模式,写出更加优雅、高效的代码。

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

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

相关文章

【卫星通信】3GPP标准提案:面向NB-IoT(GEO)场景的IMS信令优化方案-降低卫星通信场景下的语音呼叫建立时延

一、引言 随着5G非地面网络(NTN)技术的演进,基于NB-IoT的卫星通信(如GEO地球同步轨道卫星)逐渐成为偏远地区语音服务的重要补充。然而,传统IP多媒体子系统(IMS)的信令流程在带宽受限…

软件测试之简单基础的安全测试方法(另外包含软测面试题库)

文章目录 前言安全测试是什么简单基础的安全测试方法密码安全操作权限验证SQL注入xss脚本攻击文件上传下载安全漏洞扫描Web扫描APP扫描 面试题库(仅参考)参考目录 前言 阅读本文前请注意最后编辑时间,文章内容可能与目前最新的技术发展情况相…

LCEL:LangChain 表达式语言详解与测试工程师的实践指南

引言 在 AI 应用开发中,如何高效地组合多个步骤(如提示模板、模型调用、输出解析)并优化执行流程,是开发者和测试工程师共同面临的挑战。LangChain Expression Language (LCEL) 作为 LangChain 的核心功能之一,提供了…

LeetCode面试经典150题—旋转数组—LeetCode189

原题请见:Leetcode189-旋转数组 1、题目描述 2、题目分析 首先容易想到的最简单的方案,是算出来移动K步之后,新数组的每一个坐标与原坐标的映射关系,然后根据映射关系放到一个全新的数组,再把新数组的值赋给原数组。…

2.5 Rviz使用教程

新建终端,键入命令 roslaunch wpr_simulation wpb_simple.launch 再新建终端,键入命令 rviz修改Fix Frame 为 base_footprint 点击add之后选择RobotModel 再增加一个LaserScan 选择激光雷达话题 可视化效果 配置的两种方法 1.在Gazebo运行的基础上&…

基于SpringBoot+JSP开发的招投标采购信息平台

角色: 管理员、普通用户 技术: 后端:Spring Boot Mybatis-Plus MySQL 前端:JSP 核心功能: 该平台是一个用于管理投标和招标信息的系统,主要提供信息发布、用户管理和交易管理等核心功能。 功能介绍…

【项目实训#10】HarmonyOS API文档RAG检索系统后端实现

【项目实训#10】HarmonyOS API文档RAG检索系统后端实现 文章目录 【项目实训#10】HarmonyOS API文档RAG检索系统后端实现一、背景简介二、RAG技术原理与架构设计2.1 RAG技术原理回顾与提升2.2 系统架构设计 三、RAG引擎核心实现3.1 RAG引擎初始化3.2 查询向量化3.3 文档检索实现…

专注于PLC数据采集MES交互解决方案

专注于PLC数据采集MES交互解决方案 前篇文章我们讲到当下的制造行业在工业4.0的大趋势下,MES系统成为现场制造过程管制的有利武器,更是质量追踪的一把好工具。我们要知道产品在各个加工环节的结果。除了人工在各个制造环节录入制造结果外,更…

微信小程序实现文字逐行动画效果渲染显示

1. 微信小程序实现文字逐行动画效果渲染显示 在微信小程序开发中,为了文字逐行动画效果渲染可以通过JavaScript 和 WXML 的动态数据绑定来实现,实现文字逐行显示的效果,同时结合 CSS 动画提升视觉体验。   如果需要更复杂的动画效果(如缩放、移动等),可以使用微信小程序…

Redux 原理深度剖析

1. Redux 实现 定义 Action 和 Reducer 类型,为了简便,先用JavaScript来演示。 1.1. 定义Action和Reducer类型 // 定义 Action 类型 /*** typedef {Object} Action* property {string} type*/// 定义 Reducer 类型 /*** callback Reducer* param {any…

【LangChain】4 基于文档的问答

对于给定的文档, 比如从PDF、网页、公司主页中提取构建的内部文档集合,我们可以使用大语言模型来回答关于这些文档内容的问题,以帮助用户更有效地获取和使用他们所需要的信息。这种方式非常有效且灵活地适用于实际应用场景,因为它不仅仅利用大…

基于Netty的TCP Server端和Client端解决正向隔离网闸数据透传问题

背景 因为安装了正向隔离网闸&#xff0c;导致数据传输的时候仅支持TCP协议和UDP协议&#xff0c;因此需要开发TCP Client和Server服务来将数据透传&#xff0c;当前环境是获取的数据并将数据转发到kafka 1.引入依赖 <dependency><groupId>io.netty</groupId>…

Cursor链接远程服务器实现项目部署

想获取更多高质量的Java技术文章&#xff1f;欢迎访问Java技术小馆官网&#xff0c;持续更新优质内容&#xff0c;助力技术成长 技术小馆官网 在软件开发过程中&#xff0c;远程服务器开发是一种常见的工作模式。通过远程连接服务器进行代码编写和环境配置&#xff0c;可以充分…

Redis集群模式之Redis Cluster(3)

上篇文章我们讲解了Redis Cluster的状态监测与恢复过程&#xff0c;这篇文章我们来进行Redis Cluster内容的收尾&#xff0c;将其扩容和缩容的过程进行讲解&#xff0c;并分析RedisCluster的优缺点。 扩容和缩容 当集群中出现容量限制或者其他一些原因需要扩容时&#xff0c;R…

Cursor ReAct Agent技术架构

一、架构核心思想 “零熵操作交给AI”理念 Cursor通过ReAct模式实现编程中重复性工作的自动化&#xff1a; 零熵操作&#xff1a;机械性任务&#xff08;代码补全/格式化/重构/语法修复/导入管理&#xff09; Tab-away机制&#xff1a;一键接受AI建议&#xff0c;保持思维连续…

国学IP行业实战洞察:聚焦创客匠人,解锁创始人IP与知识变现新路径

国学行业正经历“文化价值”与“商业变现”的深度融合&#xff0c;2023年市场规模突破千亿大关&#xff0c;年增速超 10%。在“IP化数字化”浪潮中&#xff0c;创客匠人作为垂直领域技术服务商&#xff0c;以全链路工具矩阵为支点&#xff0c;撬动国学创始人IP从内容生产到商业…

R语言开发入门完整指南

R语言开发入门完整指南 目录 R语言简介环境配置包管理基本语法数据类型和结构数据操作统计分析数据可视化编程结构实用技巧学习资源 R语言简介 R是一种专为统计计算和图形设计的编程语言&#xff0c;广泛应用于数据分析、统计建模、机器学习和数据可视化。R语言具有以下特点…

ObservedV2装饰器和Trace装饰器

为了对嵌套类对象属性变化直接观测&#xff0c;华为提供了ObservedV2和Trace装饰器。这两个装饰器必须搭配使用&#xff0c;单独使用任何一个都不会起任何作用&#xff1b;在继承类中也可监测&#xff1b;ObservedV2的类实例目前不支持使用JSON.stringify进行序列化&#xff0c…

6月计算机新书:深度学习、大模型、DeepSeek

六月&#xff0c;这个充满活力与希望的季节&#xff0c;三本重磅新书《深度学习&#xff1a;基础与概念》、《MCP极简开发&#xff1a;轻松打造高效智能体》与《大模型应用开发&#xff1a;RAG实战课》翩然而至&#xff0c;为我们开启了一场探索科技前沿的奇妙之旅。一起来看详…

扁平风格职场商务通用PPT模版分享

扁平风格PPT模版&#xff0c;创意卡通扁平化通用PPT模版&#xff0c;创意扁平化励志论文答辩PPT模版&#xff0c;卡通职场商务PPT模版&#xff0c;职场培训&#xff0c;项目策划&#xff0c;工作总结类PPT模版&#xff0c;互联网电子商务PPT模版 扁平风格职场商务通用PPT模版分…