几种场景的单例模式思考总结(饿汉、懒汉、线程安全懒汉、反射破坏、反序列化破坏)

本文主要是介绍几种场景的单例模式思考总结(饿汉、懒汉、线程安全懒汉、反射破坏、反序列化破坏),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

今天先直接上代码,后续在补充一下单例模式的定义。。

饿汉式单例模式(线程安全):

public class Singleton {private static final Singleton instance = new Singleton();private Singleton() {}public static Singleton getInstance() {return instance;}
}

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

public class Singleton {private static Singleton instance;private Singleton() {}public static Singleton getInstance() {if (instance == null) {instance = new Singleton();}return instance;}
}

懒汉式单例模式(线程安全,双重检查锁定):

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

反射破坏单例(非线程安全):

其中反射破坏单例是指通过反射机制,可以绕过单例模式的控制,从而在单例类的构造方法中多次创建实例,破坏了单例模式的设计意图。

所以通过在单例类的构造方法中增加判断,如果已经存在实例,则抛出异常或返回已存在的实例。

public class Singleton {private static Singleton instance;private Singleton() {if (instance != null) {throw new RuntimeException("Cannot instantiate singleton class using reflection");}}public static Singleton getInstance() {if (instance == null) {instance = new Singleton();}return instance;}
}

反序列化破坏单例(线程安全):

反序列化破坏单例是指在使用序列化和反序列化机制时,可能会破坏单例模式的设计。Java 的序列化机制允许将对象转换为字节序列,以便于存储或传输,而反序列化则是将字节序列还原为对象。
**在单例类中增加 readResolve() 方法是为了在对象反序列化时返回同一实例,从而保证单例的唯一性。**这样可以防止在反序列化时创建多个实例,保持了单例模式的设计意图。
在给定的代码中,readResolve() 方法返回的是单例类的静态实例 instance,这意味着无论在何种情况下反序列化该对象,都将返回同一个实例。这样可以确保反序列化的对象与已存在的单例实例相同,避免了创建新的实例,从而保持了单例的唯一性。
需要注意的是,readResolve() 方法的访问控制符应该是 private,以确保只有单例类本身可以调用该方法,防止在外部通过反射等方式调用。

import java.io.Serializable;public class Singleton implements Serializable {private static final Singleton instance = new Singleton();private Singleton() {}public static Singleton getInstance() {return instance;}// 在反序列化过程中,readResolve() 方法会在对象反序列化时被调用,直接返回单例实例,确保单例的唯一性。protected Object readResolve() {return instance;}
}

这篇关于几种场景的单例模式思考总结(饿汉、懒汉、线程安全懒汉、反射破坏、反序列化破坏)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/867624

相关文章

Linux线程同步/互斥过程详解

《Linux线程同步/互斥过程详解》文章讲解多线程并发访问导致竞态条件,需通过互斥锁、原子操作和条件变量实现线程安全与同步,分析死锁条件及避免方法,并介绍RAII封装技术提升资源管理效率... 目录01. 资源共享问题1.1 多线程并发访问1.2 临界区与临界资源1.3 锁的引入02. 多线程案例2.1 为

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1

Java Stream流之GroupBy的用法及应用场景

《JavaStream流之GroupBy的用法及应用场景》本教程将详细介绍如何在Java中使用Stream流的groupby方法,包括基本用法和一些常见的实际应用场景,感兴趣的朋友一起看看吧... 目录Java Stream流之GroupBy的用法1. 前言2. 基础概念什么是 GroupBy?Stream

Linux系统中查询JDK安装目录的几种常用方法

《Linux系统中查询JDK安装目录的几种常用方法》:本文主要介绍Linux系统中查询JDK安装目录的几种常用方法,方法分别是通过update-alternatives、Java命令、环境变量及目... 目录方法 1:通过update-alternatives查询(推荐)方法 2:检查所有已安装的 JDK方

java如何实现高并发场景下三级缓存的数据一致性

《java如何实现高并发场景下三级缓存的数据一致性》这篇文章主要为大家详细介绍了java如何实现高并发场景下三级缓存的数据一致性,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 下面代码是一个使用Java和Redisson实现的三级缓存服务,主要功能包括:1.缓存结构:本地缓存:使

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么

在MySQL中实现冷热数据分离的方法及使用场景底层原理解析

《在MySQL中实现冷热数据分离的方法及使用场景底层原理解析》MySQL冷热数据分离通过分表/分区策略、数据归档和索引优化,将频繁访问的热数据与冷数据分开存储,提升查询效率并降低存储成本,适用于高并发... 目录实现冷热数据分离1. 分表策略2. 使用分区表3. 数据归档与迁移在mysql中实现冷热数据分

Olingo分析和实践之EDM 辅助序列化器详解(最佳实践)

《Olingo分析和实践之EDM辅助序列化器详解(最佳实践)》EDM辅助序列化器是ApacheOlingoOData框架中无需完整EDM模型的智能序列化工具,通过运行时类型推断实现灵活数据转换,适用... 目录概念与定义什么是 EDM 辅助序列化器?核心概念设计目标核心特点1. EDM 信息可选2. 智能类

Java中的xxl-job调度器线程池工作机制

《Java中的xxl-job调度器线程池工作机制》xxl-job通过快慢线程池分离短时与长时任务,动态降级超时任务至慢池,结合异步触发和资源隔离机制,提升高频调度的性能与稳定性,支撑高并发场景下的可靠... 目录⚙️ 一、调度器线程池的核心设计 二、线程池的工作流程 三、线程池配置参数与优化 四、总结:线程

Spring Boot 与微服务入门实战详细总结

《SpringBoot与微服务入门实战详细总结》本文讲解SpringBoot框架的核心特性如快速构建、自动配置、零XML与微服务架构的定义、演进及优缺点,涵盖开发环境准备和HelloWorld实战... 目录一、Spring Boot 核心概述二、微服务架构详解1. 微服务的定义与演进2. 微服务的优缺点三