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

单例模式简介

概念

单例模式(Singleton Pattern)是一种创建型设计模式,它确保一个类只有一个实例,并提供全局访问点。单例模式的核心思想是限制某个类只能创建一个对象实例,并提供对该实例的全局访问。这样可以避免多个对象间的冲突和资源浪费,同时也方便了对该实例的管理和控制。

特点

  1. 类内部负责创建唯一实例。
  2. 提供静态方法获取该唯一实例。
  3. 通过私有化构造函数防止外部直接创建新对象。
  4. 全局共享,方便统一管理。

优点

  1. 提供了对唯一实例的控制和访问方式。
  2. 避免了多个对象间资源浪费和冲突。
  3. 方便在整个应用中共享数据或状态。

缺点

  1. 违反了单一职责原则,因为该类既负责自身功能,又负责管理自己的唯一实例。
  2. 可能引起性能问题,在高并发环境下需要考虑线程安全性。

适用场景

  1. 需要保证系统中某个类只有一个实例时使用。
  2. 需要频繁地进行创建、销毁对象操作时可以减少资源消耗。

实现方式

    饿汉式(Eager Initialization

      实现原理:

  1. 在类加载时,静态变量会被初始化为默认值。
  2. 饿汉式通过将唯一实例声明为私有的静态常量,并在声明时进行初始化。
  3. 类的其他部分可以通过访问该静态常量来获取该唯一实例。

     实现代码:

public class Singleton {private static final Singleton instance = new Singleton();private Singleton() {// 私有化构造函数}public static Singleton getInstance() {return instance;}
}

上述代码中,私有化构造函数确保其他类无法直接通过new关键字创建新的实例。通过将唯一实例声明为private static final,并在静态代码块中进行初始化,可以保证该实例在类加载时就被创建。由于饿汉式在类加载时就创建了唯一实例,因此不存在线程安全问题。然而,饿汉式也存在以下问题:

  1. 可能导致资源浪费:如果该单例对象很大或者需要耗费较多资源,在应用启动阶段就创建可能会导致不必要的资源浪费。
  2. 不能延迟初始化:无法根据需要来延迟创建对象,在某些情况下可能会造成性能问题。

     3.不能处理异常情况:如果在创建过程中发生异常,则无法进行错误处理或恢复操作。

   懒汉式(Lazy Initialization

      实现原理

  1. 在类加载时,静态变量会被初始化为默认值。
  2. 懒汉式通过将唯一实例声明为私有的静态变量,并在需要时进行延迟初始化。
  3. 类的其他部分可以通过访问该静态变量来获取该唯一实例。

      实现代码:

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()方法时,只有一个线程能够进入临界区创建新的对象,其他线程则等待。

然而,懒汉式也存在以下问题:

  1. 线程安全性低:由于使用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()方法时,只有一个线程能够进入临界区创建新的对象,其他线程则等待。

双重检查锁机制可以避免每次都加锁带来的性能损耗,在第一次判断为空后才进行同步操作,提高了性能。然而,双重检查锁也存在以下问题:

  1. 可能导致指令重排:在某些情况下,由于指令重排优化的存在,可能会导致多个线程同时通过第一次检查并进入临界区创建对象。
  2. 不能处理异常情况:如果在创建过程中发生异常,则无法进行错误处理或恢复操作。

需要注意的是,在Java 5及以上版本中使用volatile关键字可以解决指令重排的问题。

   静态内部类(Static Inner Class

    实现原理:通过静态内部类来持有唯一实例,并且在需要时进行延迟初始化。

  1. 静态内部类可以访问外部类的静态成员,但不会随着外部类的加载而被加载。
  2. 当调用外部类的 getInstance() 方法时,才会触发静态内部类 SingletonHolder 的加载和初始化。
  3. 此时唯一实例被创建并赋值给外部类的 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)是一种简洁且安全的单例模式实现方式,它利用枚举类型的特性来保证只有一个实例存在。

     实现原理:

  1. 枚举类型在Java中是线程安全的,并且只能被实例化一次。
  2. 枚举类型默认提供了一个私有构造函数,该构造函数在枚举常量被初始化时调用。
  3. 在枚举类中定义一个唯一的枚举常量,即为单例对象。

实现代码:

public enum Singleton {INSTANCE;// 可以添加其他成员变量和方法public void doSomething() {// 单例对象的操作逻辑}
}

上述代码中,Singleton 是一个带有唯一常量 INSTANCE 的枚举类型。通过访问 Singleton.INSTANCE 就可以获取到该唯一实例。

这种方式能够保证线程安全性、懒加载和防止反射攻击。因为在Java中,enum 类型会自动提供序列化机制、线程安全等特性,并防止通过反射创建多个对象。因此,枚举单例是一种简洁而安全的单例模式实现方式,适用于大多数场景。

http://www.lryc.cn/news/139795.html

相关文章:

  • WPF自定义命令及属性改变处理
  • macbook m1 docker中使用go
  • 【Hello Network】DNS协议 NAT技术 代理服务器
  • Android 使用模拟器模拟Linux操作系统
  • 机器学习基础之《分类算法(5)—朴素贝叶斯算法原理》
  • # Go学习-Day6
  • 分布式 - 服务器Nginx:一小时入门系列之 HTTPS协议配置
  • 探秘Linux系统性能监控神器!Linux和Python技术持续学习者必看!
  • 文心一言续写太监小说《名侦探世界的巫师》
  • Solidity 合约安全,常见漏洞(第三篇)
  • Linux安装Redis数据库,无需公网IP实现远程连接
  • 智慧政务,长远布局——AIGC引领,加速推进数字化政府建设
  • 中央处理器(CPU):组成、指令周期、数据通路、控制方式、控制器、指令流水线,补充(多处理器系统、硬件多线程)
  • 开源微服务如何选型?Spring Cloud、Dubbo、gRPC、Istio 详细对比
  • Nginx的HTTPS部署与安全性能优化
  • 5.8. Trusted Board Boot
  • 微信小程序——van-field中的left-icon属性自定义
  • 一文学会lua脚本
  • 学习JAVA打卡第四十二天
  • 2023-8-25食物链
  • 为什么要使用IP地址进行定位
  • CSS概念
  • 淘宝API技术解析,实现关键词搜索淘宝商品(商品详情接口等)
  • Redis 7 教程 数据类型 基础篇
  • -bash: tree: command not found 的解决方法
  • SPI总线协议
  • Ubuntu20.04配置mysql配置主从复制
  • HTTP 服务器(基于go实现)
  • 【整数二分】
  • 开发一款AR导览导航小程序多少钱?ar地图微信小程序 ar导航 源码