Java设计模式之单例模式
什么是设计模式?
百科:
设计模式是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。
设计模式是软件行业的通用的设计标准,在Java同样通用,主要有23种设计模式如下:
有的小伙伴可能会问,这么多,学得完吗?
答:不好意思,不要太自信了,一般人还真学不完,不过一些常用的设计模式,例如上图中标红的单例模式、工厂模式、代理模式等设计模式,还是需要花些时间和精力去多多了解一下,相信会对自己在程序设计或写代码时有很大的帮助。
本文主要来聊一聊设计模式中创建型的单例模式,进入正文~
单例模式是什么?
学习Java的小伙伴,相信都写过Class类吧,创建某个类实例化对象的核心是new MyClass()来实现,如果没有任何设计规范,在日常开发写代码时,如果实例被用的地方很多,每次调用的时候都通过new MyClass()得到实例化对象,代码重复而且频繁的创建对象还影响性能,而有些场景我们只需要提供该类的一个实例即可,例如平时比较常见的线程池、日志对象、缓存等,一般只需要确保有一个实例即可,这种确保某个类只有一个实例并且能够类自身提供自动创建实例化对象的设计模式即称为单例模式。
单例模式设计的原则是什么?
- 构造方法私有化:既然是单例,就不能将类的构造函数暴露在外面,因此需要重写构造函数为私有化;
- 要考虑线程安全:多线程环境下,要确保不会构造出多个实例对象。
Java实现单例模式的5种方式?
关于Java实现单例模式的有几种方式,网上有很多说法,有5种、6种甚至7种实现方式,本文出于单例模式设计的两个主要原则构造方法私有化和要考虑线程安全,不考虑线程安全的其他实现方式没有任何意义,主要有5种实现方式:
懒汉
使用懒汉式写法,主要是通过synchronized修饰实例化方法getInstance,保证了线程安全,并且只有调用getInstance时才初始化,顾此得名懒汉。
懒汉写法1:
/** * 单例模式之懒汉写法1 */ public class Singleton { private static Singleton instance = null; private Singleton(){} public synchronized static Singleton getInstance(){ if (instance == null){ instance = new Singleton(); } return instance; } }
懒汉写法2:
该写法等价于写法1,原因在于关键字synchronized的灵活运用,放在方法上修饰,加锁的对象是Singleton,等效于将synchronized移到方法内部作为一个同步块,并通过括号中的Singleton.class显示指定锁对象,效果是一样的。
/** * 单例模式之懒汉写法2 */ public class Singleton { private static Singleton instance = null; private Singleton(){} public static Singleton getInstance(){ synchronized(Singleton.class) { if (instance == null) { instance = new Singleton(); } } return instance; } }
饿汉
饿汉写法,只需要定义一个static静态变量instance = new Singleton(),简单的理解为在类加载时,也会完成单例对象的实例化工作。
/** * 单例模式之饿汉 */ public class Singleton { private static Singleton instance = new Singleton(); private Singleton(){} public static Singleton getInstance(){ return instance; } }
细心的小伙伴会发现该过程并没有使用到synchronized关键字,那会不会线程不安全呢?答案是,不会,如果你大概了解过Java虚拟机即JVM(Java Virtual Machine),那你可能知道类加载过程为:加载 -> 验证 ->解析 ->初始化,而初始化阶段是执行类构造器<clinit>()方法的过程,<clinit>()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块中的语句合成产生的。
《深入理解Java虚拟机》类加载机制章节部分说明:
虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确地加锁、同步,如果多个线程同时如初始化一个类,那么只会有一个线程去执行这个类的<clinit>()方法,其他线程都需要阻塞等待,知道活动线程执行<clinit>()方法完毕。
静态内部类
静态内部类这种方式,其实就是在类的内部创建一个static SingletonInner
静态内部类,然后在静态内部类的内部再定义一个static final修饰的静态常量INSTANCE = new Singleton(),同样static修饰的SingletonInner静态内部类,会在JVM加载类时完成类的初始化并完成自己定义的静态常量单例实例化过程。
/** * 单例模式之静态内部类 */ public class Singleton { private static class SingletonInner{ private static final Singleton INSTANCE = new Singleton(); } private Singleton(){} public static Singleton getInstance(){ return SingletonInner.INSTANCE; } }
双重校验锁DCL(Double Check Lock)
DCL写法,其实与单例模式之懒汉写法2区别在于,synchronized同步块外面再套一层判断,并且使用了能确保线程安全核心volatile关键字修饰instance,表明单例变量是内存共享的,能够保证在多线程环境下的即时可见性。
/** * 单例模式之双重校验锁DCL */ public class Singleton { private volatile static Singleton instance = null; private Singleton(){} public static Singleton getInstance(){ if ( instance == null ){ synchronized (Singleton.class){ if (instance == null) instance = new Singleton(); } } return instance; } }
枚举(num)
枚举方式很容易被大家给忽略掉了,但这种方式我觉得是最简单且又友好的一种推荐创建单例的方式,通过enum修饰Singleton单例类,仅需定义一个INSTANCE,然后在静态方法实例化方法getInstance中直接返回INSTANCE即可。
/** * 单例模式之枚举 */ public enum Singleton { INSTANCE; public static Singleton getInstance(){ return INSTANCE; } }
总结
设计模式之单例模式,看似挺简单,其实还涉及了枚举enum、同步锁synchronized、JVM类加载机制、多线程volatile关键字的使用等Java的N个知识点。
本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注猪先飞的更多内容!
相关文章
- 这篇文章主要介绍了如何利用java语言实现经典《复杂迷宫》游戏,文中采用了swing技术进行了界面化处理,感兴趣的小伙伴可以动手试一试...2022-02-01
java 运行报错has been compiled by a more recent version of the Java Runtime
java 运行报错has been compiled by a more recent version of the Java Runtime (class file version 54.0)...2021-04-01- 这篇文章主要介绍了在java中获取List集合中最大的日期时间操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-08-15
- 这篇文章主要介绍了教你怎么用Java获取国家法定节假日,文中有非常详细的代码示例,对正在学习java的小伙伴们有非常好的帮助,需要的朋友可以参考下...2021-04-23
- 这篇文章主要介绍了Java如何发起http请求的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-03-31
- 说起C#和Java这两门语言(语法,数据类型 等),个人以为,大概有90%以上的相似,甚至可以认为几乎一样。但是在工作中,我也发现了一些细微的差别...2020-06-25
- 神马是“解释器模式”?先翻开《GOF》看看Definition:给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。在开篇之前还是要科普几个概念: 抽象语法树: 解释器模式并未解释如...2014-06-07
- 这篇文章主要介绍了解决Java处理HTTP请求超时的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-03-29
- 这篇文章主要为大家介绍了JavaScript设计模式中的装饰者模式,对JavaScript设计模式感兴趣的小伙伴们可以参考一下...2016-01-21
- 这篇文章主要介绍了java 判断两个时间段是否重叠的案例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-08-15
- 这篇文章主要介绍了超简洁java实现双色球若干注随机号码生成(实例代码),本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2021-04-02
- 这篇文章主要介绍了Java生成随机姓名、性别和年龄的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-10-01
java 画pdf用itext调整表格宽度、自定义各个列宽的方法
这篇文章主要介绍了java 画pdf用itext调整表格宽度、自定义各个列宽的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-01-31- 这篇文章主要介绍了java正则表达式判断前端参数修改表中另一个字段的值,需要的朋友可以参考下...2021-05-07
Java使用ScriptEngine动态执行代码(附Java几种动态执行代码比较)
这篇文章主要介绍了Java使用ScriptEngine动态执行代码,并且分享Java几种动态执行代码比较,需要的朋友可以参考下...2021-04-15- 这篇文章主要介绍了Java开发实现人机猜拳游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2020-08-03
- 这篇文章主要介绍了Java List集合返回值去掉中括号('[ ]')的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-08-29
Java中lombok的@Builder注解的解析与简单使用详解
这篇文章主要介绍了Java中lombok的@Builder注解的解析与简单使用,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2021-01-06- 下面小编就为大家带来一篇java中String类型变量的赋值问题介绍。小编觉得挺不错的。现在分享给大家,给大家一个参考。...2016-03-28
Java 8 Stream 的终极技巧——Collectors 功能与操作方法详解
这篇文章主要介绍了Java 8 Stream Collectors 功能与操作方法,结合实例形式详细分析了Java 8 Stream Collectors 功能、操作方法及相关注意事项,需要的朋友可以参考下...2020-05-20