创建型(二) - 单例模式

news/2024/11/22 20:03:27/

一、概念

单例设计模式(Singleton Design Pattern):一个类只允许创建一个对象(或者实例),那这个类就是一个单例类。

优点:在内存里只有一个实例,减少了内存的开销,避免一个全局使用的类频繁地创建与销毁。

缺点:没有接口,不能继承,与开闭职原则冲突。

使用场景:有些数据在系统中只应该保存一份,就比较适合设计为单例类。比如,系统的配置信息类。除此之外,我们还可以使用单例解决资源访问冲突的问题。

二、实现

1、步骤:

  • 构造方法定义为私有方法。
  • 定义一个私有的类静态实例。
  • 提供一个公有的获取实例的静态方法。

2、实现方式

  • 饿汉模式
public class HungryModeSingleton {private static final HungryModeSingleton mInstance = new HungryModeSingleton();private HungryModeSingleton() {}public static HungryModeSingleton getInstance() {return mInstance;}
}

总结:在类加载的时候就对实例进行初始化,没有线程安全问题;获取实例的静态方法没有使用同步,调用效率高;但是没有使用懒加载,如果该实例从始至终都没被使用过,则会造成内存浪费。

  • 懒汉模式
public class LazyModeSingleton {private static LazyModeSingleton mInstance;private LazyModeSingleton () {}public static synchronized LazyModeSingleton getInstance() {if (mInstance == null){mInstance = new LazyModeSingleton();}return mInstance;}
}

总结:上面是线程安全的懒汉模式,在第一次使用的时候才进行初始化,达到了懒加载的效果;但是,这种写法每次获取实例都要进行同步(加锁),因此在频繁获取对象的时候效率较低。

  • 双检锁/双重校验锁(DCL,即 double-checked locking)
 public class DCLModeSingleton {private static volatile DCLModeSingleton mInstance;private DCLModeSingleton () {}public static synchronized DCLModeSingleton getInstance() {if (mInstance == null){synchronized (DCLModeSingleton.class){if (mInstance == null){mInstance = new DCLModeSingleton();}}} return mInstance;      }
}      

总结:双重校验锁方式在第一次使用的时候才进行初始化,达到了懒加载的效果。并且只有第一次进行初始化才进行同步,因此不会有效率方面的问题。

在上面的例子中,DCLModeSingleton的实例使用了volatile关键字进行修饰,主要是为了解决在一些JDK低版本上有指令重排的问题。
指令重排问题:当线程A执行到9行的时候,线程B执行到6行,CPU在进行对象创建的时候,内部会在保证不影响最终结果的前提下对指令进行重新排序(不影响最终结果只是针对单线程)。例如:
创建对象步骤:分配空间-> 初始化对象 ->设置mInstance指向。
指令重排后:分配空间-> 设置mInstance指向 ->初始化对象
此时线程B执行到第6行,发现mInstance不为空,但是对象并没有初始化完成,会出现问题,但是概率较低。
使用volatile修饰符可以禁止指令重排,防止出现该问题。

  • 静态内部类
public class StaticModeSingleton {private StaticModeSingleton () {}private static class SingletonHolder{private static final StaticModeSingleton INSTANCE = new StaticModeSingleton();}public static StaticModeSingleton getInstance() {return SingletonHolder.INSTANCE;}
}

总结:SingletonHolder 是一个静态内部类,当外部类StaticModeSingleton被加载的时候,并不会创建 SingletonHolder 实例对象。只有当调用 getInstance() 方法时,SingletonHolder 才会被加载,这个时候才会创建instanceinstance 的唯一性、创建过程的线程安全性,都由 JVM 来保证。所以,这种实现方法既保证了线程安全,又能做到延迟加载。

  • 枚举
public enum Singleton {INSTANCE;//添加自己需要的操作public void doSomeThing() {}
}

总结:非懒加载,线程安全,可以避免反序列化重新创建对象。

3、实现方式优缺点对比
设计模式优缺点.PNG

三、应用场景

  • 网站计数器。
  • 应用程序的日志应用。
  • Web项目中的配置对象的读取。
  • 数据库连接池。
  • 多线程池。

单例模式详解


http://www.ppmy.cn/news/1054300.html

相关文章

中国移动秋招攻略,网申测评和面试

中国移动秋招简介 按照往年的惯例来看,移动会在每年的8月份发布相关秋招信息,紧接着考生并进行网申,面试的时间跨度也非常的长,大概是9~12月份。整个招聘流程,包括投递简历网申,笔试测评,面试录…

Cyanine3 NHS ester生物分子的标记与共价结合1032678-38-8

​欢迎来到星戈瑞荧光stargraydye!小编带您盘点: Cyanine3 NHS ester是一种荧光染料,可用于将含有游离氨基(-NH2)的生物分子如蛋白质、抗体、肽段、核酸等进行标记和共价结合。这个过程通常称为NHS酯化反应&#xff0c…

使用Matlab coder 生成函数‘referencePathFrenet’ 对应C/C++代码含有超大数组

嵌入式需要使用C/C集成,开发使用Matlab,然后使用 coder 生成函数‘referencePathFrenet’ 生成了对应的C/C代码,然而C代码含有大量超大数组导致嵌入式无法集成: 分析Matlab 源代码发现是dclothoid.m 里面路径插值的时候默认使用了…

solidity0.8.0的应用案例9:代理合约

代码由OpenZeppelin的Proxy合约简化而来。 代理模式 Solidity合约部署在链上之后,代码是不可变的(immutable)。这样既有优点,也有缺点: 优点:安全,用户知道会发生什么(大部分时候)。坏处:就算合约中存在bug,也不能修改或升级,只能部署新合约。但是新合约的地址与…

野狼测评工作室:电视机顶盒哪个牌子好?电视机顶盒排行榜

欢迎各位来到野狼测评,本期工作室的测评产品是电视机顶盒,电视机顶盒不仅可以让老电视变得智能化,也可以让配置落后的智能电视在不换新的前提下升级配置,本周我们测评了十几款热门电视机顶盒,盘点电视机顶盒排行榜&…

区块链-Web3.0-什么是Web3.0?

一、什么是Web 3.0 Web 3.0,也被称为“去中心化Web”或“智能Web”,是互联网的下一代,它使用了分布式系统技术、区块链技术和智能合约等新型技术,旨在构建一个更加去中心化、安全、透明和智能的互联网。Web 3.0 可以带来更广泛的…

微服务框架之SpringBoot面试题汇总

微服务框架之SpringBoot面试题汇总 什么是Spring Boot? 多年来,随着新功能的增加,spring变得越来越复杂。Spring项目,我们必须添加构建路径或添加Maven依赖关系,配置应用程序服务器,添加spring配置。因此&…

LeetCode150道面试经典题-- 求算数平方根(简单)

1.题目 给你一个非负整数 x ,计算并返回 x 的 算术平方根 。 由于返回类型是整数,结果只保留 整数部分 ,小数部分将被 舍去 。 注意:不允许使用任何内置指数函数和算符,例如 pow(x, 0.5) 或者 x ** 0.5 。 2.示例 …