在Java中实现完整的单例模式

在Java中实现完整的单例模式个人认为单例模式是设计模式中最简单也是最常用的一种 是对有限资源合理利用的一种方式 这个模式看似简单 但是其中蕴含了关于并发 类加载 序列化等一系列深层次的知识 如果理解不够深 就有可能在高并发时遇到难以预期的异常 或者会造成资源浪费

个人认为单例模式是设计模式中最简单也是最常用的一种,是对有限资源合理利用的一种方式。这个模式看似简单,但是其中蕴含了关于并发、类加载、序列化等一系列深层次的知识,如果理解不够深,就有可能在高并发时遇到难以预期的异常,或者会造成资源浪费。

所以本文会从将目前Java领域最常用的几种单例模式列出来,供大家参考。

WHAT

维基百科给出了解释、实现的思路以及应该注意的地方:

单例模式,也叫单子模式,是一种常用的软件设计模式,属于创建型模式的一种。在应用这个模式时,单例对象的类必须保证只有一个实例存在。

实现单例模式的思路是:一个类能返回对象一个引用(永远是同一个)和一个获得该实例的方法(必须是静态方法,通常使用getInstance这个名称);当我们调用这个方法时,如果类持有的引用不为空就返回这个引用,如果类保持的引用为空就创建该类的实例并将实例的引用赋予该类保持的引用;同时我们还将该类的构造函数定义为私有方法,这样其他处的代码就无法通过调用该类的构造函数来实例化该类的对象,只有通过该类提供的静态方法来得到该类的唯一实例。

单例模式在多线程的应用场合下必须小心使用。如果当唯一实例尚未创建时,有两个线程同时调用创建方法,那么它们同时没有检测到唯一实例的存在,从而同时各自创建了一个实例,这样就有两个实例被构造出来,从而违反了单例模式中实例唯一的原则。 解决这个问题的办法是为指示类是否已经实例化的变量提供一个互斥锁(虽然这样会降低效率)。

类图是:

singleton pattern

WHY

正如定义所说,单例模式就是整个内存模型中,只有一个实例。实例少了,内存占用就少。同时,只有一个实例,也就只需要构建一个对象,计算就少。对于构造过程中需要大量计算或者占用大量资源的对象,只创建一次,就减少了资源占用和内存占用。

HOW

饿汉式

饿汉式是最简单的一种实现,在类装载过程中,完成实例化,避免多线程问题。

实现一:静态实例参数与静态代码块

public class EagerSingleton { 
    private static final EagerSingleton INSTANCE = new EagerSingleton(); private EagerSingleton() { 
    } public static EagerSingleton getInstance() { 
    return INSTANCE; } } 

根据java的特性,饿汉式还可以变种写法,有的地方称为静态代码块方式:

public class EagerSingleton { 
    private static EagerSingleton INSTANCE = null; static { 
    INSTANCE = new EagerSingleton(); } private EagerSingleton() { 
    } public static EagerSingleton getInstance() { 
    return INSTANCE; } } 

这两种方式只是在写法上的区别,优缺点没有区别,只是借助Java语言特性的不同写法,所以归为一类。

饿汉式有两个明显的缺点:

  1. 类装载过程即完成实例化,如果整个应用生命周期内,实例没有使用,也就是浪费资源了。
  2. 因为没有办法向构造函数传递不同的参数,如果需要通过个性化参数定制实例时,这种方式就不支持了。

实现二:静态内部类

针对饿汉式第一个缺点,我们可以借助静态内部类的方式,将对象实例化的时间延后。

public class EagerSingleton { 
    private EagerSingleton() { 
    } private static class EagerSingletonInstance { 
    private static final EagerSingleton INSTANCE = new EagerSingleton(); } public static EagerSingleton getInstance() { 
    return EagerSingletonInstance.INSTANCE; } } 

但是,依然不能很好的解决第二个缺点,如果需要根据不同的参数实现不同的实例,可以采用下面说的懒汉式实现。

懒汉式

懒汉式比饿汉式的一个优点,就是能够在使用的时候再进行实例化。但是,馅饼总是要伴随着陷阱,懒汉式写法有更多的坑,一不小心就摔着了。

错误一:单线程实现

public class LazySingleton { 
    private static LazySingleton INSTANCE = null; private LazySingleton() { 
    } public static LazySingleton getInstance() { 
    if (INSTANCE == null) { 
    INSTANCE = new LazySingleton(); } return INSTANCE; } } 

之所以定义为单线程实现,是因为INSTANCE == null这个判断,一个线程通过这个判断,开始进行对象实例化,但是还没有实例化完成,另一个线程又来了,这个时候,对象还没有实例化,就也会开始进行实例化,造成不必要的浪费。

错误二:同步方法

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

这种方式解决了多线程的问题,但是也引入了新的性能问题:太慢。synchronized把整个方法包起来,也就是每个线程进入的时候,都需要等待其他线程结束调用,才能拿到实例,在性能敏感的场景,是比较致命的。

错误三:同步代码块之单次检查

public class LazySingleton { 
    private static LazySingleton INSTANCE = null; private LazySingleton() { 
    } public static LazySingleton getInstance() { 
    if (INSTANCE == null) { 
    synchronized (LazySingleton.class) { 
    INSTANCE = new LazySingleton(); } } return INSTANCE; } } 

这种写法看似将同步代码缩小,但也缩小了多线程保障,也犯了第一种写法的错误,属于没有对多线程有基本了解写出的低级错误代码。

错误四:同步代码块之双重检查

public class LazySingleton { 
    private static LazySingleton INSTANCE = null; private LazySingleton() { 
    } public static LazySingleton getInstance() { 
    if (INSTANCE == null) { 
    synchronized (LazySingleton.class) { 
    if (INSTANCE == null) { 
    INSTANCE = new LazySingleton(); } } } return INSTANCE; } } 

这种写法在一定程度上属于正确的写法,双重判断可以很好的实现线程安全和延迟加载。如果到这里就结束,那就是谬以千里的毫厘之差。

双重检查和同步代码块都没有问题,问题出在INSTANCE = new LazySingleton()这句话。在JVM中,为了充分利用CPU计算能力,会进行重排序优化,INSTANCE = new LazySingleton()做了三件事:

  1. 为 INSTANCE 初始化栈空间
  2. 为 LazySingleton 分配内存空间,实例化对象
  3. INSTANCE 指向 LazySingleton 实例分配的内存空间

因为重排序优化的存在,真正执行的过程中,可能会出现1-2-3的顺序,也可能出现1-3-2的顺序。如果是1-3-2,INSTANCE 指向了 LazySingleton 实例分配的内存空间后,就不是null,另外一个线程进入判断null时,就会直接返回 INSTANCE,但是这个时候 LazySingleton 实例化还没有完成,就可能出现意想不到的异常。

正确:双重检查+阻止重排序

public class LazySingleton { 
    private static volatile LazySingleton INSTANCE = null; private LazySingleton() { 
    } public static LazySingleton getInstance() { 
    if (INSTANCE == null) { 
    synchronized (LazySingleton.class) { 
    if (INSTANCE == null) { 
    INSTANCE = new LazySingleton(); } } } return INSTANCE; } } 

这种写法比上面那种,就差在volatile这个关键字。

枚举

懒汉式和饿汉式都能够适用于多线程并发场景,但是通过反序列化或反射可以实例化对象,这样依然不能满足单例模式的要求,所以可以借助枚举实现,枚举可以完美避免多线程并发问题,而且可以防止反序列化和反射创建新对象。第一次看到这样定义单例模式,是在《Effective Java》中,多读经典书还是挺好的。

public enum EnumSingleton { 
    INSTANCE; public void method1() { 
    // do something } public Object method2() { 
    // do something and return something else return new Object(); } } 

在开发实践中,枚举可以满足绝大部分场景,而且写法简单,定义单例的逻辑只需要三行代码,简洁而不简单,三行代码可以保证线程安全。同时枚举的反序列化只是通过name查找对象,不会产生新的对象;根据JVM规范,通过反射创建枚举对象时,会抛出IllegalArgumentException异常。这样,相当于通过语法糖防止反序列化和反射破坏单例。

场景

  1. 无状态工具类:这种工具类不需要记录状态,只保证正确的应用就行,可以通过单例模式来定义。
  2. 数据共享:即多个不相关的两个线程或者进程之间实现通信。因为是一个实例,如果它的属性或者变量值被修改,所有引用都是同时修改的,当然需要 volatile 来定义变量。比如网站的计数器。
  3. 日志应用:通常应用会向日志文件写日志信息,为了实时向文件写,通常会使用单例模式,保证有一个实例持有文件,然后进行操作。
  4. 数据库连接池:数据库连接是一种数据库资源,使用数据库连接池,主要是节省打开或者关闭数据库连接所引起的效率损耗,这种效率上的损耗还是非常昂贵的,通过单例模式来维护,就可以大大降低这种损耗。
  5. Web应用的配置对象:读取文件需要消耗时间,如果读取大文件,消耗的时间和资源更久,所以通过单例模式可以大大降低消耗。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/213110.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月18日 下午6:32
下一篇 2026年3月18日 下午6:32


相关推荐

  • 记jxbrowser异常

    记jxbrowser异常异常 com teamdev jxbrowser chromium BrowserExcep IPCstartupfa 处理 结束 Xfvb 进程 重新创建 并重新配置环境变量 ps ef grepXvfb grep vgrepnohupXv 2 screen01024x gt dev null2 gt amp 1 amp echo exportDISPLA 2 gt gt bashrc

    2025年10月18日
    4
  • patch文件的执行和制作

    patch文件的执行和制作执行 patch 文件 br patch p NUM0 1 4 patchbr nbsp br NUM setting p0givestheen p1givesbr nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp u howard src blurfl blurfl cbr nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp withoutthele p4givesbr nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp blurfl blurfl c 制作 p

    2025年11月15日
    6
  • Maven菜鸟教程:Maven仓库的分类[通俗易懂]

    Maven菜鸟教程:Maven仓库的分类[通俗易懂]Maven菜鸟教程中有很多知识点需要大家学习,Maven仓库分类就是其中一点,为了更具体地增加对仓库的理解,这里详细介绍一下各种仓库。本地仓库Maven在根据坐标查找依赖的构件时,先是在本地仓库中查找。默认情况下,不管是Windows操作系统还是Linux操作系统,每个用户在自己的用户目录下都有一个路径名为.m2/repository/的目录,这个目录就是Maven的本地仓库目录。比如,笔者的用户名是Noble,计算机上的默认本地仓库的目录就是C:\Users\Noble.m2

    2025年10月8日
    4
  • 文心一言开源模型部署实战与性能深度测评

    文心一言开源模型部署实战与性能深度测评

    2026年3月12日
    3
  • Maximal Information Coefficient (MIC)最大互信息系数详解与实现「建议收藏」

    Maximal Information Coefficient (MIC)最大互信息系数详解与实现「建议收藏」MICMIC即:MaximalInformationCoefficient最大互信息系数。使用MIC来衡量两个基因之间的关联程度,线性或非线性关系,相较于MutualInformation(MI)互信息而言有更高的准确度。MIC是一种优秀的数据关联性的计算方式。本篇文章将会详细介绍MIC的算法原理,优缺点以及Python的具体实现方式,并给出一个可视化方案。互信息?互信息(Mut…

    2026年1月17日
    7
  • 使用git新建分支推送项目

    使用git新建分支推送项目前言 一个还在努力的编程小白转载请标注来源 git 分支操作一 新建自己的分支二 推送项目到仓库三 错误四 参考一 新建自己的分支如果单纯的提交分支 不用 pull 原来其他人的分支首先在项目文件上进行 git 初始化 gitinit 出现该文件说明初始化成功然后我们去 github 获取项目仓库 http 连接 gitremoteadd 你的 http 查看本地的连接 gitremote v 首先创建分支 gitcheckout b

    2026年3月16日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号