饿汉式在类加载时创建实例,线程安全但不支持懒加载;2. 懒汉式在首次调用时创建实例,支持懒加载但线程不安全;3. 线程安全的懒汉式通过synchronized实现线程安全和懒加载,但性能较低;4. 双重检查锁通过volatile和同步块实现高效线程安全与懒加载,实现较复杂;5. 静态内部类利用类加载机制实现线程安全和懒加载,推荐使用;6. 枚举实现线程安全且防止反射和序列化攻击,但不支持懒加载;防御反射攻击可在构造方法中检查实例是否存在并抛出异常,枚举单例天然防止反射攻击;单例模式通过类加载机制或同步手段保证多线程环境下的线程安全;单例模式限制一个类仅有一个实例并提供全局访问点,而工厂模式封装对象创建过程,两者关注点不同,单例控制实例数量,工厂抽象创建逻辑,工厂模式可创建单例对象,但二者本质不同,完整结束。
Java单例模式的核心在于确保一个类只有一个实例,并提供一个全局访问点。实现这一点,需要控制类的实例化过程。
解决方案
Java中实现单例模式有多种方法,各有优劣,选择哪种取决于具体场景的需求,比如是否需要懒加载、线程安全等。
立即学习“Java免费学习笔记(深入)”;
1. 饿汉式(Eager Initialization)
这是最简单的实现方式,在类加载时就完成了实例化。
public class Singleton { private static final Singleton instance = new Singleton(); private Singleton() { // 私有构造方法,防止外部实例化 } public static Singleton getInstance() { return instance; } public void doSomething() { System.out.println("Singleton is doing something..."); } }
2. 懒汉式(Lazy Initialization)
在第一次调用
getInstance()
public class Singleton { private static Singleton instance; private Singleton() { // 私有构造方法,防止外部实例化 } public static Singleton getInstance() { if (instance == null) { instance = new Singleton(); } return instance; } public void doSomething() { System.out.println("Singleton is doing something..."); } }
3. 线程安全的懒汉式(Synchronized Lazy Initialization)
为了解决懒汉式的线程安全问题,可以对
getInstance()
public class Singleton { private static Singleton instance; private Singleton() { // 私有构造方法,防止外部实例化 } public static synchronized Singleton getInstance() { if (instance == null) { instance = new Singleton(); } return instance; } public void doSomething() { System.out.println("Singleton is doing something..."); } }
getInstance()
4. 双重检查锁(Double-Checked Locking)
一种更高效的线程安全懒汉式实现。
public class Singleton { private volatile static Singleton instance; private Singleton() { // 私有构造方法,防止外部实例化 } public static Singleton getInstance() { if (instance == null) { synchronized (Singleton.class) { if (instance == null) { instance = new Singleton(); } } } return instance; } public void doSomething() { System.out.println("Singleton is doing something..."); } }
volatile
5. 静态内部类(Static Inner Class)
利用类加载机制保证线程安全和懒加载。
public class Singleton { private Singleton() { // 私有构造方法,防止外部实例化 } private static class SingletonHolder { private static final Singleton instance = new Singleton(); } public static Singleton getInstance() { return SingletonHolder.instance; } public void doSomething() { System.out.println("Singleton is doing something..."); } }
6. 枚举(Enum)
最简洁的实现方式,可以防止反射攻击和序列化问题。
public enum Singleton { INSTANCE; public void doSomething() { System.out.println("Singleton is doing something..."); } }
单例模式中的反射攻击如何防御?
反射机制允许在运行时检查和修改类的行为。对于单例模式,这意味着可以通过反射创建多个实例,破坏单例的特性。防御反射攻击的主要方法是在构造方法中进行检查,如果已经存在实例,则抛出异常。例如:
public class Singleton { private static Singleton instance; private Singleton() { if (instance != null) { throw new IllegalStateException("Singleton instance already exists."); } } public static Singleton getInstance() { if (instance == null) { synchronized (Singleton.class) { if (instance == null) { instance = new Singleton(); } } } return instance; } }
这样,即使通过反射调用构造方法,也会因为抛出异常而无法创建新的实例。枚举单例天然防止反射攻击,因为Java语言规范禁止通过反射创建枚举实例。
单例模式在多线程环境下如何保证线程安全?
多线程环境是单例模式需要重点考虑的场景。线程安全意味着在多个线程同时访问单例对象时,不会出现数据不一致或者创建多个实例的情况。上述的几种实现方式中,饿汉式、静态内部类和枚举天生就是线程安全的。懒汉式需要进行同步处理,才能保证线程安全。双重检查锁是一种常用的优化手段,可以减少同步的开销。
单例模式与工厂模式的区别是什么?
单例模式和工厂模式是两种不同的设计模式,虽然它们都涉及到对象的创建,但解决的问题和应用场景不同。单例模式关注的是如何保证一个类只有一个实例,并提供全局访问点。工厂模式关注的是如何将对象的创建过程封装起来,使得客户端不需要知道具体类的创建细节。简单来说,单例模式是限制类的实例化数量,而工厂模式是抽象类的实例化过程。工厂模式可以创建单例对象,但单例模式不一定需要通过工厂模式来创建。
以上就是java怎样实现单例模式保证对象唯一 java单例模式实现的基础操作教程的详细内容,更多请关注php中文网其它相关文章!
java怎么学习?java怎么入门?java在哪学?java怎么学才快?不用担心,这里为大家提供了java速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 //m.sbmmt.com/ All Rights Reserved | php.cn | 湘ICP备2023035733号