单例模式的理解和实践

news2024/12/27 1:38:17

        在软件开发中,设计模式是开发者在特定情境下,对常见问题的通用解决方案。这些模式帮助开发者以更高效、可维护的方式编写代码。其中,单例模式(Singleton Pattern)是一种创建型设计模式,它确保一个类只有一个实例,并提供一个全局访问点。本文将详细讲解单例模式的理解及其在Java中的实践。

一、单例模式的理解

1.1 定义

单例模式是一种确保某个类只有一个实例,并提供一个全局访问点的设计模式。它适用于以下场景:

  • 资源控制:如数据库连接池、文件系统等,需要控制资源的使用,避免频繁创建和销毁对象。
  • 共享状态:如配置管理器、线程池等,需要在多个地方共享同一个状态。

1.2 关键点

  • 私有构造函数:防止外部通过new关键字创建实例。
  • 静态实例:在类内部维护一个静态实例。
  • 全局访问点:提供一个静态方法,用于返回该实例。

1.3 实现步骤

  1. 私有化构造函数:防止外部通过构造函数创建实例。
  2. 创建静态实例:在类内部定义一个静态变量,并初始化为类的实例。
  3. 提供静态方法:提供一个公共的静态方法,用于返回该实例。

二、单例模式的实践

        接下来,我们将通过具体的Java代码示例,演示如何实现单例模式。

2.1 饿汉式

        饿汉式在类加载时就创建实例,因此是线程安全的,但可能会造成资源浪费,如果实例从未被使用过。

// 饿汉式单例模式
public class SingletonEager {
    // 在类加载时就创建实例
    private static final SingletonEager INSTANCE = new SingletonEager();

    // 私有化构造函数
    private SingletonEager() {
        // 初始化代码
    }

    // 提供全局访问点
    public static SingletonEager getInstance() {
        return INSTANCE;
    }
}


优点

  • 线程安全,因为实例在类加载时就创建。

缺点

  • 即使实例未被使用,也会创建实例,可能造成资源浪费。

2.2 懒汉式(线程不安全)

懒汉式在第一次使用时才创建实例,节省资源,但线程不安全。

// 懒汉式单例模式(线程不安全)
public class SingletonLazyUnsafe {
    // 静态实例,初始为null
    private static SingletonLazyUnsafe instance;

    // 私有化构造函数
    private SingletonLazyUnsafe() {
        // 初始化代码
    }

    // 提供全局访问点
    public static SingletonLazyUnsafe getInstance() {
        if (instance == null) {
            instance = new SingletonLazyUnsafe();
        }
        return instance;
    }
}


优点

  • 延迟加载,节省资源。

缺点

  • 线程不安全,多个线程同时访问时,可能会创建多个实例。

2.3 懒汉式(线程安全)

通过同步方法或同步代码块,确保线程安全。

同步方法

// 懒汉式单例模式(线程安全,同步方法)
public class SingletonLazySafeSyncMethod {
    private static SingletonLazySafeSyncMethod instance;

    private SingletonLazySafeSyncMethod() {
        // 初始化代码
    }

    // 同步方法,确保线程安全
    public static synchronized SingletonLazySafeSyncMethod getInstance() {
        if (instance == null) {
            instance = new SingletonLazySafeSyncMethod();
        }
        return instance;
    }
}


同步代码块

// 懒汉式单例模式(线程安全,同步代码块)
public class SingletonLazySafeSyncBlock {
    private static SingletonLazySafeSyncBlock instance;
    private static final Object LOCK = new Object();

    private SingletonLazySafeSyncBlock() {
        // 初始化代码
    }

    // 同步代码块,提升性能
    public static SingletonLazySafeSyncBlock getInstance() {
        if (instance == null) {
            synchronized (LOCK) {
                if (instance == null) {
                    instance = new SingletonLazySafeSyncBlock();
                }
            }
        }
        return instance;
    }
}


优点

  • 延迟加载,节省资源。
  • 线程安全,通过同步确保只有一个实例。

缺点

  • 同步方法性能较差,因为每次调用都需要同步。
  • 同步代码块虽然性能有所提升,但仍然需要双重检查(Double-Checked Locking),代码相对复杂。

2.4 静态内部类(推荐)

静态内部类方式结合了饿汉式和懒汉式的优点,既实现了延迟加载,又保证了线程安全。

// 静态内部类单例模式(推荐)
public class SingletonStaticInnerClass {
    // 私有构造函数
    private SingletonStaticInnerClass() {
        // 初始化代码
    }

    // 静态内部类,负责创建实例
    private static class Holder {
        private static final SingletonStaticInnerClass INSTANCE = new SingletonStaticInnerClass();
    }

    // 提供全局访问点
    public static SingletonStaticInnerClass getInstance() {
        return Holder.INSTANCE;
    }
}


优点

  • 延迟加载,节省资源。
  • 线程安全,JVM保证静态内部类在第一次使用时才加载。
  • 无需同步,性能较好。

缺点

  • 实现相对复杂,需要理解静态内部类的加载机制。

2.5 枚举(最安全)

枚举单例模式是最简单、最安全的实现方式,天生防止反射和序列化攻击。

// 枚举单例模式(最安全)
public enum SingletonEnum {
    INSTANCE;

    // 提供其他方法
    public void doSomething() {
        // 实现代码
    }
}


优点

  • 防止反射攻击,因为JVM禁止通过反射机制创建枚举实例。
  • 防止序列化攻击,因为枚举的序列化机制由JVM保证,不会创建新的实例。
  • 线程安全,由JVM保证。

缺点

  • 实现简单,但可能不适合所有场景,特别是需要继承其他类的场景(Java枚举不能继承非枚举类)。

三、单例模式的注意事项

防止反射攻击:通过私有构造函数和异常处理,防止通过反射机制创建实例。

private Singleton() {
    if (instance != null) {
        throw new RuntimeException("Use getInstance() method to get the single instance of this class.");
    }
    // 初始化代码
}

防止序列化攻击:通过实现readResolve方法,防止通过序列化机制创建实例。

private Object readResolve() throws ObjectStreamException {
    return INSTANCE;
}

线程安全:确保在多线程环境下,只有一个实例被创建。

延迟加载:如果实例创建开销较大,考虑使用懒汉式或静态内部类方式。

总结

        单例模式是一种确保某个类只有一个实例,并提供一个全局访问点的设计模式。它通过私有化构造函数、创建静态实例和提供全局访问点来实现。在Java中,有多种实现方式,包括饿汉式、懒汉式(线程不安全、线程安全)、静态内部类和枚举。每种方式都有其优缺点,开发者应根据具体场景选择合适的实现方式。同时,需要注意防止反射和序列化攻击,确保线程安全。通过合理使用单例模式,可以提高代码的可维护性和性能。

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

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

相关文章

node.js基础学习-url模块-url地址处理(二)

前言 前面我们创建了一个HTTP服务器,如果只是简单的http://localhost:3000/about这种链接我们是可以处理的,但是实际运用中一般链接都会带参数,这样的话如果我们只是简单的判断链接来分配数据,就会报404找不到链接。为了解决这个问…

思科网络设备常用命令整理

思科网络设备的配置命令非常丰富,广泛应用于路由器、交换机和其他网络设备的管理与配置。以下是一些常见的思科设备配置命令,按照功能分类,以帮助你快速查找和使用。 一、基本命令 查看当前配置和状态 show running-config:查看…

2024年信号处理与神经网络应用(SPNNA 2024)

会议官网:www.spnna.org 会议时间:2024年12月13-15日 会议地点:中国武汉

Leecode经典题3-删除排序数组中的重复项

删除排序数组中的重复项 题目描述: 给你一个 非严格递增排列 的数组 nums ,请你 原地 删除重复出现的元素,使每个元素 只出现一次 ,返回删除后数组的新长度。元素的 相对顺序 应该保持 一致 。然后返回 nums 中唯一元素的个数。 …

无人机数据处理系统:原理与核心系统

一、数据处理系统的运行原理 数据获取:无人机在飞行过程中,通过搭载的传感器(如相机、激光雷达等)采集到各种类型的数据,例如图像、点云等。这些数据是后续处理和分析的基础。 数据传输:采集到的数据会通…

ElasticSearch学习篇19_《检索技术核心20讲》搜推广系统设计思想

目录 主要是包含搜推广系统的基本模块简单介绍,另有一些流程、设计思想的分析。 搜索引擎 基本模块检索流程 查询分析查询纠错 广告引擎 基于标签倒排索引召回基于向量ANN检索召回打分机制:非精确打分精准深度学习模型打分索引精简:必要的…

【尚筹网】五、管理员维护

【尚筹网】五、管理员维护 任务清单分页管理管理员信息目标思路代码引入 PageHelperAdminMapper 中编写 SQL 语句AdminMapper 接口生成方法AdminServiceAdminHandler页面显示主体在页面上使用 Pagination 实现导航条 关键词查询页面上调整表单在翻页时保持关键词查询条件 单条删…

MySQL 启动失败问题分析与解决方案:`mysqld.service failed to run ‘start-pre‘ task`

目录 前言1. 问题背景2. 错误分析2.1 错误信息详解2.2 可能原因 3. 问题排查与解决方案3.1 检查 MySQL 错误日志3.2 验证 MySQL 配置文件3.3 检查文件和目录权限3.4 手动启动 MySQL 服务3.5 修复 systemd 配置文件3.6 验证依赖环境 4. 进一步优化与自动化处理结语 前言 在日常…

Apache storm UI如何更换默认8080端口

在搭建Apache storm环境的时候,遇到Apache storm UI默认端口是8080,但是这个端口会被其他java程序占用,导致Apache storm UI服务无法启动。报错Exception in thread “main” java.lang.RuntimeException: java.io.IOException: Failed to bi…

FPGA实现串口升级及MultiBoot(十)串口升级SPI FLASH实现

本文目录索引 工程架构example9工程设计Vivado设计Vitis设计example9工程验证1、读取FLASH ID2、擦除整个FLASH3、Blank-Check4、烧写Golden区位流5、读取FLASH内容6、烧写MultiBoot区位流(升级位流)7、MultiBoot区位流(升级位流)启动example10工程设计Vivado设计Vitis设计exam…

图解人工智能:从规则到深度学习的全景解析

🌟作者简介:热爱数据分析,学习Python、Stata、SPSS等统计语言的小高同学~🍊个人主页:小高要坚强的博客🍓当前专栏:Python之机器学习🍎本文内容:图解人工智能:…

Binder架构

一、架构 如上图,binder 分为用户层和驱动层两部分,用户层有客户端(Client)、服务端(Server)、服务管理(ServiceManager)。 从用户空间的角度,使用步骤如下(…

基于springboot中小型制造企业质量管理系统源码和论文

信息数据从传统到当代,是一直在变革当中,突如其来的互联网让传统的信息管理看到了革命性的曙光,因为传统信息管理从时效性,还是安全性,还是可操作性等各个方面来讲,遇到了互联网时代才发现能补上自古以来的…

Flutter 权限申请

这篇文章是基于permission_handler 10.2.0版本写的 前言 在App开发过程中我们经常要用到各种权限,我是用的是permission_handler包来实现权限控制的。 pub地址:https://pub.dev/packages/permission_handler permission_handler 权限列表 变量 Androi…

MATLAB期末复习笔记(下)

五、数据和函数的可视化 1.MATLAB的可视化对象 图形对象是 MATLAB用来创建可视化数据的组件。每个对象都有一个名为句柄 的唯一标识符。使用该句柄,您可以通过设置对象 属性 来操作现有图形对象的特征 ROOT: :即电脑屏幕 Figure :图窗…

web安全从0到1:burp-suite3

声明! 学习视频来自B站up主 **泷羽sec** 有兴趣的师傅可以关注一下,如涉及侵权马上删除文章,笔记只是方便各位师傅的学习和探讨,文章所提到的网站以及内容,只做学习交流,其他均与本人以及泷羽sec团队无关&a…

深度学习:梯度下降法

损失函数 L:衡量单一训练样例的效果。 成本函数 J:用于衡量 w 和 b 的效果。 如何使用梯度下降法来训练或学习训练集上的参数w和b ? 成本函数J是参数w和b的函数,它被定义为平均值; 损失函数L可以衡量你的算法效果&a…

Linux:文件系统inode

早期,存储文件的设备是磁盘(当下的市场几乎都是SSD),但大家习惯的把它们都称为磁盘,磁盘是用来表示区分内存的存储设备。而在操作系统看来,这个存储设备的结构就是一个线性结构,这一点很重要。 …

94.【C语言】解析预处理(2)

目录 1.带有副作用的宏参数 代码 一个判断最大值的宏代码 运行结果 分析 "副作用"的解释 2.宏替换规则 解释 3.宏和函数的对比 附一张对比表 承接93.【C语言】解析预处理(1)文章 1.带有副作用的宏参数 代码 一个判断最大值的宏代码 #define MAX(a, b) (…

Linux学习笔记12 systemd的其他命令

前文已经介绍了systemd在系统初始化中起到的作用和服务的管理和配置。这里补充一下systemd的其他工具和系统进程的管理 前文 Linux学习笔记10 系统启动初始化,服务和进程管理(上)-CSDN博客 Linux学习笔记11 系统启动初始化,服务…