设计模式——单例模式

这篇具有很好参考价值的文章主要介绍了设计模式——单例模式。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

单例模式

定义

确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。

即保证一个类只有一个实例,并且提供一个全局访问点

优缺点、应用场景

优点

  1. 单例对象在内存中只有一个实例,减少了内存的开支。尤其对于一个频繁创建、销毁的对象时,单例模式的优势就更明显。
  2. 减少系统的性能开销。一个对象的创建需要占用较多资源(例如:读取配置信息、产生其他依赖)时,可以在系统启动时产生单例对象,再通过永久驻留内存的方式来解决。
  3. 避免资源的多重占用。例如一个写文件的操作,单例模式可以避免对同一文件的同时写操作。

缺点

  1. 单例模式一般没有接口,拓展困难。
  2. 对测试不利,严格创建单例的环境中,只能在单例对象创建后才能进行测试。
  3. 单例模式与单一职责原则有冲突。

场景

  1. 重量级的对象,不需要多个实例,如线程池,数据库连接池。

实现方式

  • 懒汉模式
  • 饿汉模式
  • 静态内部类
  • 枚举类型

懒汉模式(外部类写法)

单线程下,只需要创建一次instance对象即可

设计模式——单例模式,设计模式,设计模式,单例模式
设计模式——单例模式,设计模式,设计模式,单例模式

多线程下,就有可能出现同时创建实例

设计模式——单例模式,设计模式,设计模式,单例模式
设计模式——单例模式,设计模式,设计模式,单例模式

解决方法:synchronized

设计模式——单例模式,设计模式,设计模式,单例模式

优化(synchronized+双重非空校验)

虽然可以进行同步,但是并不是每一次都需要对来访的对象进行加锁,只有尝试创建时才需要加锁

/**
 * 懒汉模式
 */
class LazyMan{
	private volatile static LazyMan instance;
	private LazyMan(){
	}

	public static LazyMan getInstance() {
		if (instance == null){
			// 如果两个以上线程检测到instance为null,则竞争一把类锁
			synchronized (LazyMan.class){
				if (instance == null){
					instance = new LazyMan();
				}
			}
		}
		return instance;
	}
}

反编译查看new的过程

步骤:
对.java文件进行:javac操作,得到.class文件
再对.class文件进行:javap -v操作
设计模式——单例模式,设计模式,设计模式,单例模式

  1. 首先在堆空间创建该类的引用
  2. 将引用的内存地址复制到栈内存,压到栈顶
  3. 初始化构造方法(这里是无参构造方法)
  4. 将引用从栈中弹出
  5. 给对象赋值

重排序问题

根据反编译的步骤:分配空间、初始化、引用赋值,

这三步中的初始化和引用赋值是可以调换位置的。

但是如果赋值发生在初始化之前,则有可能出现空指针异常。所以要使用volatile,让线程到主存中访问数据,这样就不会出现null了。
设计模式——单例模式,设计模式,设计模式,单例模式

懒汉模式小结

设计模式——单例模式,设计模式,设计模式,单例模式

饿汉模式

初始化阶段就创建好一个对象,其他对象要访问,就只能访问这个对象。
本质是依赖JVM的类加载机制,确保实例的唯一性
设计模式——单例模式,设计模式,设计模式,单例模式

饿汉模式小结

设计模式——单例模式,设计模式,设计模式,单例模式

根据JVM的类加载过程:
其中准备过程是根据基本类型和对象类型进行初始化,基本类型如Integer就为0,String类型就为null。

而饿汉模式则是在JVM的初始化阶段唯一的对变量赋值,确保了对象的唯一。

静态内部类实现单例模式

设计模式——单例模式,设计模式,设计模式,单例模式

即通过静态内部类的方式创建唯一实例,不提供公有的构造函数
并且只能通过公有的getInstance()方法获取私有对象

静态内部类实现小结

设计模式——单例模式,设计模式,设计模式,单例模式

反射攻击

通过资源类的反射,获取到私有构造方法的使用权,创建实例。
设计模式——单例模式,设计模式,设计模式,单例模式
结果返回false

解决方法一

在私有构造中判断instance是否已经创建,进行锁死条件,防止反射攻击
设计模式——单例模式,设计模式,设计模式,单例模式

解决方法二

枚举法

根据反射newInstance 的源码,可以发现如果反射的类是添加了枚举enum类型的,则不允许创建该对象。

会抛出非法参数的异常
设计模式——单例模式,设计模式,设计模式,单例模式
设计模式——单例模式,设计模式,设计模式,单例模式
设计模式——单例模式,设计模式,设计模式,单例模式
可以证明,枚举类型的反射是不允许创建对象的

序列化攻击

根据序列化的特点及其内部的实现原理,序列化与反序列化不会经过我们所指定的方法。所以可以通过序列化来进行攻击。
设计模式——单例模式,设计模式,设计模式,单例模式
返回false,即序列化前后对象不一致,反序列化后再创建了一个对象。

解决方法:Serializable

重写方法readResolve(),并且设置序列化版本

设计模式——单例模式,设计模式,设计模式,单例模式
设计模式——单例模式,设计模式,设计模式,单例模式
设计模式——单例模式,设计模式,设计模式,单例模式
设计模式——单例模式,设计模式,设计模式,单例模式

这样就确保了序列化前后的对象是同一个。

枚举类的序列化问题

枚举类不存在序列化攻击的问题,跟反射攻击一样

根据ObjectInputStream类中提供的方法,可以发现枚举类型在进行反序列化时被加载到了类加载器中,收到JVM的保护。

不可变的类型
设计模式——单例模式,设计模式,设计模式,单例模式
设计模式——单例模式,设计模式,设计模式,单例模式
设计模式——单例模式,设计模式,设计模式,单例模式文章来源地址https://www.toymoban.com/news/detail-589911.html

到了这里,关于设计模式——单例模式的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 设计模式(单例模式)

            保证指定的类只有一个实例,不能创建出其他的实例                 1.1 代码展示                 1.2 Singleton类中instance对象的创建时机                 Singleton类中instance对象的创建时机:在Singleton类被jvm加载的时候创建,Singleton类会在第一次使用的时

    2024年02月15日
    浏览(51)
  • 设计模式-单例模式

          单例模式(Singleton Pattern)是设计模式中最简单且最常用的一种创建型模式,其目的是保证一个类在整个系统中只存在一个实例,并提供全局访问点来获取这个唯一实例。这种模式主要适用于那些需要频繁实例化然后又希望避免因为多次实例化而消耗过多资源或产生副

    2024年01月17日
    浏览(53)
  • 【设计模式-单例模式】

    在一个项目中的全局范围内, 一个类 有且仅有一个实例对象 。这个唯一的实例对象给其他模块提供数据的 全局访问 。这样的模式就叫 单例模式 。 单例模式的典型例子就是任务队列。 首先, 考虑单例模式的要求为有且仅有一个实例对象。那么就先从构造函数入手。类的构

    2024年02月13日
    浏览(59)
  • 设计模式——单例模式

    确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。 即保证一个类只有一个实例,并且提供一个全局访问点 优点 单例对象在内存中只有一个实例,减少了内存的开支。尤其对于一个频繁创建、销毁的对象时,单例模式的优势就更明显。 减少系统的性能

    2024年02月16日
    浏览(62)
  • 设计模式 ~ 单例模式

    单例模式:指在确保一个类只有一个实例,创建之后缓存以便继续使用,并提供一个全局访问点来访问该实例; 前端对于单例模式不常用,但对于单例的思想无处不在; 如:弹窗、遮罩层、登录框、vuex redux 中的 store; 闭包: 模块化:

    2024年02月16日
    浏览(57)
  • 设计模式-单例模式进阶

    在前面的文章(设计模式-单例模式)中,我们分别介绍了四种单例设计模式,包括 普通恶汉式单例、双重检查锁单例(DCL)、静态内部类单例以及枚举单例 。但是,这四种模式还有一些 问题 我们没有仔细分析,以至于我们无法深入分析他们的优点以及可能存在的问题,更无法确

    2024年02月16日
    浏览(38)
  • 设计模式 : 单例模式笔记

    一个类 只能创建一个对象 ,这样的类的设计模式就称为单例模式,该模式保证 系统中 该类 只能有一个实例 (并且 父子进程共享 ),一个很典型的单例类就是C++STL的内存池 C++单例模式的基本设计思路: 私有化 构造函数 ,删除默认的 拷贝构造函数 和 赋值运算符重载 防止对象被直

    2024年02月12日
    浏览(52)
  • 设计模式一(单例模式)

    主要思路:将构造方法私有化,并对外提供一个static的方法来创建对象 缺点:一开始就创建对象,占用系统资源 单线程下不会出现问题,但多线程会会有并发问题,main方法的测试结果: 会发生同一时间创建了多个对象,所以出现了DCL双重检索 可以实现延迟实例化,并且是

    2024年01月23日
    浏览(43)
  • 设计模式_单例模式

    保证该类只有一个实例( static私有变量 ),并提供一个访问的它的全部访问点( getInstance() 方法 ),该单例可以被所有程序模块共享. 1)此类不可被复制. 2)此类不可被公开构造. 也就是说,在c++中,它的 构造函数,拷贝构造函数,赋值函数 不能被公开调用. 个人理解,“懒汉式”,那一定很

    2024年02月12日
    浏览(43)
  • 设计模式篇---单例模式

    单例模式是结构最简单的设计模式,通过单例模式可以保证在整个系统中的一个类只有一个实例,从而节约系统资源。举个例子,比如windows电脑下的任务管理器只能打开一个,这个就是单例模式,如果不这样做,则会弹出多个窗口,但这些窗口显示的内容完全一样,浪费了资

    2024年02月10日
    浏览(30)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包