设计模式之创建型设计模式(一):单例模式 原型模式

news2025/1/4 16:34:21

单例模式 Singleton

1、什么是单例模式

在软件设计中,单例模式是一种创建型设计模式,其主要目的是确保一个类只有一个实例,并提供一个全局访问点。

这意味着无论何时需要该类的实例,都可以获得相同的实例,而不会创建新的对象。

单例模式通常用于控制对资源的访问,例如配置文件、数据库连接,或者共享的实例。

2、为什么使用单例模式

  1. 资源共享:单例模式可以确保在整个应用程序中只有一个实例,从而节省系统资源,避免多次创建相同对象。
  2. 全局访问:通过单例模式,可以在任何需要时轻松访问该类的实例,而无需传递它作为参数。
  3. 懒加载:单例模式可以实现懒加载,即只有在需要时才创建实例,而不是在应用程序启动时就创建。

3、如何实现单例模式

示例场景:设计实现一个 Logger 类,用于记录应用程序中的日志信息。

非单例模式的实现
public class Logger {
    private List<String> logs = new ArrayList<>();

    public void log(String message) {
        logs.add(message);
    }
}

// 在应用程序的不同部分创建Logger实例
Logger logger1 = new Logger();
Logger logger2 = new Logger();

logger1.log("Message from logger1");
logger2.log("Message from logger2");

System.out.println(logger1.getLogs());  // ['Message from logger1']
System.out.println(logger2.getLogs());  // ['Message from logger2']
单例模式的实现
public class Logger {
    private static Logger instance;  // 单例实例
    private List<String> logs = new ArrayList<>();

    private Logger() {}

    public static Logger getInstance() {
        if (instance == null) {
            instance = new Logger();
        }
        return instance;
    }

    public void log(String message) {
        logs.add(message);
    }

    public List<String> getLogs() {
        return logs;
    }
}

// 在应用程序的不同部分获取Logger实例
Logger logger1 = Logger.getInstance();
Logger logger2 = Logger.getInstance();

logger1.log("Message from logger1");
logger2.log("Message from logger2");

System.out.println(logger1.getLogs());  // ['Message from logger1', 'Message from logger2']
System.out.println(logger2.getLogs());  // ['Message from logger1', 'Message from logger2']
代码对比说明
  1. 资源共享:使用单例模式后,Logger 实例在整个应用程序中是唯一的,确保了日志的全局共享,避免了每次创建实例都生成新对象的问题。
  2. 全局访问:单例模式允许在应用程序的任何地方访问相同的 Logger 实例,而不必传递它。这提高了代码的简洁性和可维护性。
  3. 懒加载:单例模式的实现中,通过 getInstance 方法进行了懒加载,只有在实例不存在时才创建。这确保了在应用程序启动时不会预先创建实例,而是在需要时才进行创建,提高了效率。

4、是否存在缺陷和不足

  1. 全局状态:单例模式引入了全局状态,可能会导致代码的耦合性增加。由于单例在整个应用程序中都是可见的,其他部分可能难以预测它的状态,从而使代码难以理解和维护。
  2. 并发控制:在多线程环境中,需要考虑并发控制的问题。当多个线程同时尝试第一次获取单例实例时,可能会导致创建多个实例。为了解决这个问题,需要引入同步机制,但这会带来性能的开销。
  3. 隐藏依赖关系:使用单例模式可能导致隐藏了类之间的依赖关系,因为它可以在任何地方访问单例实例。这使得代码的结构变得不够清晰,降低了模块化的优势。
  4. 单一职责原则破坏:单例模式通常负责两个职责控制实例化过程和提供全局访问点。这可能违反了单一职责原则,使得代码的职责不够清晰。
  5. 测试困难:由于单例模式创建实例的逻辑通常包含在类内部,很难进行单元测试。测试过程可能会依赖于全局状态,导致测试的不确定性。

5、如何缓解缺陷和不足

  1. 使用依赖注入:考虑使用依赖注入来解决单例模式引入的全局状态问题。通过将依赖项传递给需要它们的类,可以更好地控制类之间的关系。
  2. 懒加载与双重检查锁定:在多线程环境中,可以使用懒加载和双重检查锁定等技术来确保实例的唯一性,并提高性能。
  3. 考虑其他创建型模式:根据具体需求,考虑其他创建型设计模式,如工厂方法模式或建造者模式,以避免单例模式的一些限制。
  4. 避免过度使用:单例模式并不适用于所有情况,避免过度使用。在确保有明确需求时使用,而不是为了方便而滥用。

虽然单例模式存在一些缺点,但在许多情况下,它仍然是一种有效的设计选择。在使用单例模式时,需要仔细权衡其优点和缺点,并根据具体情况选择合适的设计方式。

原型模式 Prototype

1、什么是原型模式

原型模式是一种创建型设计模式,核心思想是通过复制现有对象来创建新对象,而不是通过实例化类,这种创建新对象的方式更加高效,尤其当新对象的创建成本较高时,使用原型模式可以提高性能。

2、为什么使用原型模式

  1. 性能提升:当新对象的创建成本较高时,通过复制现有对象可以避免重复执行初始化和配置的开销,提高对象创建的效率。
  2. 灵活性:原型模式可以在运行时动态配置对象,通过克隆可以得到一个与原始对象相似的新对象,而无需修改结构。
  3. 简化对象创建:原型模式隐藏了对象的创建细节,使得对象的创建更加简单和统一。

3、如何实现原型模式

非原型模式的实现
public class Book {
    private String title;
    private String author;

    public Book(String title, String author) {
        this.title = title;
        this.author = author;
    }

    // Getters and setters...

    public Book clone() {
        return new Book(this.title, this.author);
    }
}
原型模式的实现
public class Book implements Cloneable {
    private String title;
    private String author;

    public Book(String title, String author) {
        this.title = title;
        this.author = author;
    }

    // Getters and setters...

    @Override
    public Book clone() {
        try {
            return (Book) super.clone();
        } catch (CloneNotSupportedException e) {
            return null;
        }
    }
}

4、是否存在缺陷和不足

  1. 深克隆问题:默认的 clone 方法是浅克隆,即只克隆对象本身,而不克隆其引用类型的成员变量。如果对象包含引用类型的成员变量,可能需要进行深克隆,以避免共享引用对象。
  2. 构造函数不执行:在使用原型模式时,对象的构造函数不会执行,这可能导致某些初始化逻辑未被执行。

5、如何缓解缺陷和不足

  1. 深克隆的实现:如果需要深克隆,可以通过重写 clone 方法,手动克隆引用类型的成员变量,确保新对象独立于原始对象。
  2. 初始化方法:在原型对象中提供一个初始化方法,可以在克隆对象后手动调用该方法,以确保必要的初始化逻辑得以执行。

通过以上缓解措施,可以更好地应对深克隆和构造函数不执行等问题,使得原型模式更加灵活和实用,在实际应用中,根据具体需求和场景选择是否使用原型模式。

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

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

相关文章

API接口能力不足?Bug处理慢?Lazada开放平台API商品接入

7月30日正式发布的Lazada开放平台2.0&#xff08;Lazada Open Platform 2.0&#xff09;&#xff0c;从商品API、订单API、IM&#xff08;即时通信&#xff09; API、营销工具等几大方向&#xff0c;带来全新升级的API体系&#xff0c;共新增47个接口、优化19个接口&#xff0c…

JVM-9-Class类文件的结构

Java技术能够一直保持着非常良好的向后兼容性&#xff0c;Class文件结构的稳定功不可没。 Class文件是一组以8个字节为基础单位的二进制流&#xff0c;各个数据项目严格按照顺序紧凑地排列在文件之中。 Class文件格式采用一种类似于C语言结构体的伪结构来存储数据&#xff0c…

详细教程 - 从零开发 鸿蒙harmonyOS应用 第八节——鸿蒙操作系统中的文件读写操作封装

一、引言 鸿蒙操作系统是华为自主研发的全场景操作系统。在这篇博客中&#xff0c;我们将探讨如何在鸿蒙操作系统中实现文件读写操作的封装。 二、文件读写操作 在鸿蒙操作系统中&#xff0c;文件读写操作是一个常见的需求。下面是一个简单的文件读写操作的封装示例&#xff1…

数据分析场景下,企业大模型选型的思路与建议

来源/作者&#xff1a;爱分析 随着大模型带来能力突破&#xff0c;让AI与数据分析相互结合&#xff0c;使分析结果更好支撑业务&#xff0c;促进企业内部数据价值释放&#xff0c;成为了当下企业用户尤为关注的话题。本次分享主要围绕数据分析场景下大模型底座的选型思路&#…

上传xml文件进行跨站脚本攻

文件路径或者URL&#xff1a;xxxxx <?xml version"1.0" encoding"iso-8859-1"?> <xsl:stylesheet version"1.0" xmlns:xsl"http://www.w3.org/1999/XSL/Transform"> <xsl:template match"/"> <html…

部署threestudio | stable zero123

选择我在autodl的stable-zero123这个镜像&#xff0c;或者直接选这个基础环境 开机后切换到conda的base环境 这里注意一点就是目前stable-zero123这个镜像还没解决的问题&#xff0c;就是没法使用xformers&#xff0c;所以如果重新配置&#xff0c;在这里就要先安装指定版本的…

liunx下用C++使用freetype库在opencv上打中文字

1、/visualizer.cpp:11:10: fatal error: ft2build.h: 没有那个文件或目录 11 | #include <ft2build.h> freetype安装问题&#xff0c;要把文件拉到根目录&#xff0c;不然找不到文件 2、编译失败找不到定义 /usr/bin/ld: CMakeFiles/interactive_face_detection_de…

树莓派Ubuntu系统安装OpenCV教程

硬件&#xff1a;树莓派4B 2G 内存卡32G 软件&#xff1a;系统是ubuntu-mate-20.04.1-desktop-arm64raspi.img 步骤如下&#xff1a;一、更新系统软件包列表和安装基本工具 sudo apt-get update sudo apt-get upgrade sudo apt-get install build-essential cmake git二、安装…

【Linux】Linux运维基础

Linux简介&#xff1a; Linux是一个开源的操作系统内核&#xff0c;最初由Linus Torvalds创建。它通常与GNU工具一起使用&#xff0c;以创建一个完整的操作系统。Linux操作系统有许多基于内核的发行版&#xff0c;如Ubuntu、CentOS、Debian等&#xff0c;每个发行版都有其独特的…

解决docker alpine /bin/sh: ./main: not found

解决docker alpine /bin/sh: ./main: not found golang中编译之后的二进制文件部署在alpine镜像中出现了not found问题解决这种情况是因为动态链接库位置错误导致的&#xff0c;alpine镜像使用的是musl libc而不是gun libc。因而动态链接库的位置不一致。在基础镜像内执行&…

13. 从零用Rust编写正反向代理, HTTP中的压缩gzip,deflate,brotli算法

wmproxy wmproxy是由Rust编写&#xff0c;已实现http/https代理&#xff0c;socks5代理&#xff0c; 反向代理&#xff0c;静态文件服务器&#xff0c;内网穿透&#xff0c;配置热更新等&#xff0c; 后续将实现websocket代理等&#xff0c;同时会将实现过程分享出来&#xff…

IDEA版SSM入门到实战(Maven+MyBatis+Spring+SpringMVC) -Spring的AOP前奏

第一章 AOP前奏 1.1 代理模式 代理模式&#xff1a;我们需要做一件事情&#xff0c;又不期望自己亲力亲为&#xff0c;此时&#xff0c;可以找一个代理【中介】 我们【目标对象】与中介【代理对象】不能相互转换&#xff0c;因为是“兄弟”关系 1.2 为什么需要代理【程序中…

MQTT中的遗嘱和证明(Last Will and Testament, LWT)

在MQTT中&#xff0c;遗嘱和证明&#xff08;Last Will and Testament, LWT&#xff09;是一个很有用的功能&#xff0c;允许客户端指定一条消息&#xff0c;当非预期的断开连接发生时&#xff0c;由代理&#xff08;broker&#xff09;代替它们自动发布出去。提供了一个可靠通…

Redis原理之网络模型笔记

目录 1. 阻塞IO 2. 非堵塞IO 3. IO多路复用 ​3.1 select 3.2 poll 3.3 epoll 4. 信号驱动IO 5. 异步IO 6. Redis是单线程还是多线程 Redis采用单线程模型&#xff0c;这意味着一个Redis服务器在任何时刻都只会处理一个请求。Redis的网络模型涉及到阻塞I/O&#xff08;Blo…

基于双目RGB图像和图像深度信息的三维室内场景建模matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 4.1 双目视觉原理 4.2 深度信息获取 4.3 表面重建 5.算法完整程序工程 1.算法运行效果图预览 2.算法运行软件版本 matlab2022a 3.部分核心程序 .....................................…

快速处理EDI数据映射:知行EDI Profiler 操作指南

一个完整的EDI项目通常由建立传输通道&#xff0c;处理数据映射以及集成内部业务系统三部分组成。对用户而言&#xff0c;基于知行之桥EDI系统进行自主实施最大的挑战便是处理数据映射。EDI报文读不懂&#xff0c;映射关系太复杂……这些问题给企业造成困扰的同时也阻挡了自主实…

【Linux】进程状态、进程优先级和进程切换

> 作者简介&#xff1a;დ旧言~&#xff0c;目前大二&#xff0c;现在学习Java&#xff0c;c&#xff0c;c&#xff0c;Python等 > 座右铭&#xff1a;松树千年终是朽&#xff0c;槿花一日自为荣。 > 目标&#xff1a;了解冯诺依曼体系结构与操作系统&#xff0c;掌握…

RLC防孤岛负载测试:电力系统安全运行的重要保障

在电力系统中&#xff0c;孤岛效应是一个严重的问题&#xff0c;它可能导致电力系统的不稳定甚至崩溃。为了确保电力系统的安全运行&#xff0c;必须进行RLC防孤岛负载测试。RLC防孤岛负载测试是一种模拟电网故障后&#xff0c;对电力系统进行检测的方法&#xff0c;主要用于检…

css实现边框彩虹跑马灯效果

效果展示 代码实战 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" content"widthdevice-…

Ubuntu20.04 下编译安装 ffmpeg 和 ffplay

Ubuntu20.04 下编译安装 ffmpeg 和 ffplay 一、下载源码包二、安装依赖库三、编译四、添加环境变量五、验证是否成功六、问题 一、下载源码包 1.1 官方下载链接&#xff1a;http://ffmpeg.org/download.html 最新版本为6.1&#xff0c;点击 Download Source Code下载即可 &…