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

Java中实现单例模式的方式

1. 使用静态内部类实现单例模式

在Java中,使用静态内部类实现单例模式是一种常见而又有效的方式。这种方式被称为“静态内部类单例模式”或者“Holder模式”。这种实现方式有以下优点:

  • 懒加载(Lazy Initialization):静态内部类不会在外部类加载的时候就被加载,而是在第一次被调用的时候才会加载。这就实现了懒加载,即在需要的时候才创建单例对象,避免了在应用启动时就创建对象的开销。

  • 线程安全(Thread-Safe):由于静态内部类只会被加载一次,因此在加载过程中,其他线程无法访问静态内部类的加载过程。这就天然地保证了线程安全性。

public class Singleton {// 私有化构造函数,防止外部直接实例化private Singleton() {}// 静态内部类,用于持有单例对象private static class SingletonHolder {private static final Singleton INSTANCE = new Singleton();}// 获取单例对象的方法public static Singleton getInstance() {return SingletonHolder.INSTANCE;}
}

2. 使用序列化和反序列化实现单例模式

在Java中,使用标准的序列化和反序列化机制时,可能会破坏单例模式,因为反序列化会创建一个新的对象。为了避免这个问题,可以在单例类中添加一个特殊的方法,以便在反序列化时返回现有的单例对象。

import java.io.*;public class Singleton implements Serializable {private static final long serialVersionUID = 1L;// 私有化构造函数,防止外部直接实例化private Singleton() {}// 单例实例private static final Singleton INSTANCE = new Singleton();// 获取单例对象的方法public static Singleton getInstance() {return INSTANCE;}// 在反序列化时,确保返回相同的单例对象protected Object readResolve() throws ObjectStreamException {return INSTANCE;}public static void main(String[] args) {// 序列化try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("singleton.ser"))) {oos.writeObject(Singleton.getInstance());} catch (IOException e) {e.printStackTrace();}// 反序列化try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream("singleton.ser"))) {Singleton deserializedInstance = (Singleton) ois.readObject();System.out.println("Original Singleton HashCode: " + Singleton.getInstance().hashCode());System.out.println("Deserialized Singleton HashCode: " + deserializedInstance.hashCode());} catch (IOException | ClassNotFoundException e) {e.printStackTrace();}}
}

readResolve() 是一个特殊的回调方法,用于在反序列化时返回实际要返回的对象。这个方法允许开发者指定在反序列化过程中应该返回哪个对象。

3. 使用静态代码块实现单例模式

使用静态代码块实现单例模式是一种常见的方式。通过在静态代码块中进行对象的实例化,可以保证在类加载的时候只执行一次,从而实现单例模式。

public class Singleton {// 私有化构造函数,防止外部直接实例化private Singleton() {}// 单例实例private static final Singleton INSTANCE;// 静态代码块,在类加载时执行,用于初始化单例实例static {try {INSTANCE = new Singleton();} catch (Exception e) {throw new RuntimeException("Error creating singleton instance");}}// 获取单例对象的方法public static Singleton getInstance() {return INSTANCE;}public static void main(String[] args) {// 获取单例对象Singleton singleton = Singleton.getInstance();}
}

4. 使用枚举实现单例模式

在Java中,使用枚举(enum)实现单例模式是一种简洁且线程安全的方式。枚举本身就天然地具有单例的特性,保证在任何情况下都只有一个实例。这是因为Java中的枚举类型是线程安全且只能被实例化一次的。

  1. JVM保证枚举的线程安全性: Java中的枚举类型在JVM层面被设计为线程安全的,这是因为枚举实例在类加载的时候就被创建,且JVM会保证每个枚举值只被实例化一次。
  2. 保证实例的唯一性: 枚举类型的实例是在枚举类加载时被创建的,而且在整个应用程序的生命周期中,它们只会被创建一次。这就保证了枚举实例的唯一性,因此枚举天生就符合单例模式的要求。
  3. 简洁且防御反射攻击: 使用枚举实现单例模式的代码非常简洁,而且防御了一些反射攻击,因为枚举的实例在类加载的时候就被创建,不会受到反射的影响。
public enum Singleton {INSTANCE; // 单例实例// 可以在枚举中添加其他方法public void someMethod() {// 实现方法逻辑}public static void main(String[] args) {// 获取单例对象Singleton singleton = Singleton.INSTANCE;// 调用其他方法singleton.someMethod();}
}
http://www.lryc.cn/news/255905.html

相关文章:

  • Vue3-01-创建项目
  • Go 语言中的反射机制
  • [leetcode 前缀和]
  • Python与ArcGIS系列(十五)根据距离抓取字段
  • YOLOv8分割训练及分割半自动标注
  • jsp页面通过class或者id获取a标签上的属性的值
  • 题目:美丽的区间(蓝桥OJ 1372)
  • 解决:During handling of the above exception, another exception occurred
  • 计算机基础知识65
  • Python开发运维:Python垃圾回收机制
  • ros2/ros安装ros-dep||rosdep init错误
  • 《深入理解计算机系统》学习笔记 - 第四课 - 机器级别的程序
  • 云原生(Cloud Native)——概念,技术,背景,优缺点,实践例子
  • ElasticSearch之线程池
  • StoneDB-8.0-V2.2.0 企业版正式发布!性能优化,稳定性提升,持续公测中!
  • 【数据结构 — 排序 — 插入排序】
  • 物联网后端个人第十四周总结
  • 在uniapp中,可以使用那些预定义的样式类
  • mybatis的数据库连接池
  • Vue 的 el-select 下拉选项中,只有当文字超出时才显示提示框,未超出的则不显示
  • 【Python】pptx文件转pdf
  • response应用及重定向和request转发
  • CentOS常用基础命令大全(linux命令)2
  • 分析阿里巴巴的微服务依赖图和性能
  • Linux——基本指令(一)
  • 虚幻学习笔记10—C++函数与蓝图的通信
  • 无重复字符的最长子串(LeetCode 3)
  • 交付《啤酒游戏经营决策沙盘》的项目
  • 油猴(Tampermonkey)浏览器插件简单自定义脚本开发
  • BGP综合