吉林建设网站,全球中文网站排名,淄博网站制作定制升级,wordpress主题破解下载概念#xff1a;
单例模式#xff08;Singleton Pattern#xff09;是一种创建型设计模式#xff0c;它确保一个类只有一个实例#xff0c;并提供全局访问点。单例模式的核心思想是限制某个类只能创建一个对象实例#xff0c;并提供对该实例的全局访问。这样可以避免多个…概念
单例模式Singleton Pattern是一种创建型设计模式它确保一个类只有一个实例并提供全局访问点。单例模式的核心思想是限制某个类只能创建一个对象实例并提供对该实例的全局访问。这样可以避免多个对象间的冲突和资源浪费同时也方便了对该实例的管理和控制。
特点
类内部负责创建唯一实例。提供静态方法获取该唯一实例。通过私有化构造函数防止外部直接创建新对象。全局共享方便统一管理。
优点
提供了对唯一实例的控制和访问方式。避免了多个对象间资源浪费和冲突。方便在整个应用中共享数据或状态。
缺点
违反了单一职责原则因为该类既负责自身功能又负责管理自己的唯一实例。可能引起性能问题在高并发环境下需要考虑线程安全性。
适用场景
需要保证系统中某个类只有一个实例时使用。需要频繁地进行创建、销毁对象操作时可以减少资源消耗。
实现方式 饿汉式Eager Initialization 实现原理
在类加载时静态变量会被初始化为默认值。饿汉式通过将唯一实例声明为私有的静态常量并在声明时进行初始化。类的其他部分可以通过访问该静态常量来获取该唯一实例。 实现代码
public class Singleton {private static final Singleton instance new Singleton();private Singleton() {// 私有化构造函数}public static Singleton getInstance() {return instance;}
}上述代码中私有化构造函数确保其他类无法直接通过new关键字创建新的实例。通过将唯一实例声明为private static final并在静态代码块中进行初始化可以保证该实例在类加载时就被创建。由于饿汉式在类加载时就创建了唯一实例因此不存在线程安全问题。然而饿汉式也存在以下问题
可能导致资源浪费如果该单例对象很大或者需要耗费较多资源在应用启动阶段就创建可能会导致不必要的资源浪费。不能延迟初始化无法根据需要来延迟创建对象在某些情况下可能会造成性能问题。 3.不能处理异常情况如果在创建过程中发生异常则无法进行错误处理或恢复操作。 懒汉式Lazy Initialization 实现原理
在类加载时静态变量会被初始化为默认值。懒汉式通过将唯一实例声明为私有的静态变量并在需要时进行延迟初始化。类的其他部分可以通过访问该静态变量来获取该唯一实例。 实现代码
public class Singleton {private static Singleton instance;private Singleton() {// 私有化构造函数}public static synchronized Singleton getInstance() {if (instance null) {instance new Singleton();}return instance;}
}上述代码中getInstance()方法使用了synchronized关键字来保证线程安全。当多个线程同时调用getInstance()方法时只有一个线程能够进入临界区创建新的对象其他线程则等待。
然而懒汉式也存在以下问题
线程安全性低由于使用synchronized关键字保证了线程安全性在高并发环境下可能导致性能问题。 2.双重检查锁失效在某些情况下双重检查锁机制可能失效导致多个对象被创建。 双重检查锁Double-Checked Locking 实现原理在懒汉式Lazy Initialization的基础上增加了一次实例非空检查。 实现代码
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;}
}上述代码中使用了synchronized关键字和volatile关键字来保证线程安全性和可见性。当多个线程同时调用getInstance()方法时只有一个线程能够进入临界区创建新的对象其他线程则等待。
双重检查锁机制可以避免每次都加锁带来的性能损耗在第一次判断为空后才进行同步操作提高了性能。然而双重检查锁也存在以下问题
可能导致指令重排在某些情况下由于指令重排优化的存在可能会导致多个线程同时通过第一次检查并进入临界区创建对象。不能处理异常情况如果在创建过程中发生异常则无法进行错误处理或恢复操作。
需要注意的是在Java 5及以上版本中使用volatile关键字可以解决指令重排的问题。 静态内部类Static Inner Class 实现原理通过静态内部类来持有唯一实例并且在需要时进行延迟初始化。
静态内部类可以访问外部类的静态成员但不会随着外部类的加载而被加载。当调用外部类的 getInstance() 方法时才会触发静态内部类 SingletonHolder 的加载和初始化。此时唯一实例被创建并赋值给外部类的 instance 变量。 实现代码
public class Singleton {private Singleton() {// 私有化构造函数}private static class SingletonHolder {private static final Singleton INSTANCE new Singleton();}public static Singleton getInstance() {return SingletonHolder.INSTANCE;}
}上述代码中SingletonHolder 是一个私有的静态内部类在该内部类中持有了唯一实例 INSTANCE。当调用 getInstance() 方法时直接返回了 SingletonHolder 内的 INSTANCE 实例。
这种方式能够保证懒加载和线程安全性。因为在第一次调用 getInstance() 方法获取单例对象之前并不会触发对于SingletonHolder 类的初始化操作只有在真正需要获取单例对象时才会触发 SingletonHolder 类的加载和实例化。
静态内部类方式解决了双重检查锁存在的指令重排问题并且没有加锁操作因此性能较高。同时也能够保证线程安全性和懒加载特性。因此静态内部类方式是一种常用而有效的单例模式实现方式。 枚举单例Enum Singleton 枚举单例Enum Singleton是一种简洁且安全的单例模式实现方式它利用枚举类型的特性来保证只有一个实例存在。 实现原理
枚举类型在Java中是线程安全的并且只能被实例化一次。枚举类型默认提供了一个私有构造函数该构造函数在枚举常量被初始化时调用。在枚举类中定义一个唯一的枚举常量即为单例对象。
实现代码
public enum Singleton {INSTANCE;// 可以添加其他成员变量和方法public void doSomething() {// 单例对象的操作逻辑}
}上述代码中Singleton 是一个带有唯一常量 INSTANCE 的枚举类型。通过访问 Singleton.INSTANCE 就可以获取到该唯一实例。
这种方式能够保证线程安全性、懒加载和防止反射攻击。因为在Java中enum 类型会自动提供序列化机制、线程安全等特性并防止通过反射创建多个对象。因此枚举单例是一种简洁而安全的单例模式实现方式适用于大多数场景。