关于单例模式的N种实现方式
在开发中经常用到单例模式,单例模式也算是设计模式中最容易理解,也是最容易手写代码的模式,所以也常作为面试题来考。所以想总结一下单例模式的理论知识,方便同学们面试使用。
单例模式实现的方式只有两种类型,一种是饿汉式(类加载时就初始化)、一种是懒汉式(类加载时不初始化)。饿汉式没什么可讲究的因为它既简单也线程安全,如果条件允许一般我们都会直接用饿汉式;唯独比较麻烦的是懒汉式,考虑到线程安全,使用懒汉式的单例模式,就有多种实现方式了。
一、饿汉式
如上所述,饿汉式很简单,实现方式如下:
public class Singleton{
private static final Singleton instance = new Singleton();
private Singleton(){}
public static Singleton getInstance(){
return instance;
}
}
这种单例模式,很简单而且还安全,可以说近乎完美。它唯一的缺陷就是,这种实现方式就无法适用于单例还需要一定的配置或者传参的场景。
二、懒汉式
懒汉式的单例模式,由于要考虑到线程安全的情况,有多种实现方式。
0、如下的实现方式是一种不安全的实现方式
public class Singleton {
private static Singleton instance;
private Singleton (){}
public static Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
之所以把定义为编号0,就是除非你非常确定没有多线程的场景,因为当有多个线程并行调用 getInstance() 的时候,就会创建多个实例。所以绝大部分场景下,这种实现方式都是不应该使用。
1、低性能的懒汉式
public static synchronized Singleton getInstance() {
private static Singleton instance;
private Singleton (){}
if (instance == null) {
instance = new Singleton();
}
return instance;
}
这种实现方式,解决了多线程实例问题,但是由于直接在方法上使用synchronized关键字,即类锁同步,使得只能使用每次调用都要进行同步操作,性能比较低。
2、双重检验锁的懒汉式
public class Singleton {
private volatile static Singleton instance;
private Singleton() {
}
public static Singleton getSingleton() {
if (instance == null) { // null 检测
synchronized (Singleton.class) { // 同步检测
if (instance == null) {
instance = new Singleton();
}
}
}
return instance;
}
}
我们看到最多的情况就是没有加volatile的情形,其实你不加这个关键字,面试你的人应该也不会纠结于这个,因为99.99%的情况都是好的(要知道那些提供后台服务的,一般只会保证99%可靠性……)。之所以加是因为JVM存在指令重排的优化,导致
new Singleton() 不是一个原子操作行为,new Singleton() 这句话执行的过程大概分为以下abc步骤:
a, 给 instance 分配内存;
b, 调用 Singleton 的构造函数来初始化成员变量;
c, 将instance对象指向分配的内存空间(执行完这步 instance 就为非 null 了);
JVM 的即时编译器中存在指令重排序的优化,执行顺序可能是 abc 也可能是 acb。如果是acb,在执行到c,被线程X抢占了,这时 instance 已经是非 null 了(但却没有初始化),所以线程X会直接返回 instance,使用一个没有初始化的实例产生错误是必然的。那为什么要加volatile呢?一般我们只用到volatile的可见性功能,即对一个volatile变量的读,总是能看到(任意线程)对这个volatile变量最后的写入,而且经常用于解决同步问题。其实volatile还有一个重要的功能——禁止指令重排优化,也就是volatile标记的变量不会被编译器优化。如上所示,加上volatile以后,读操作一定会发生在abc或者acb之后,并且这个顺序是固定。
3、静态内部类实现懒汉式
public class Singleton {
private static class InternalSingleton {
private static final Singleton INSTANCE = new Singleton();
}
private Singleton (){}
public static final Singleton getInstance() {
return InternalSingleton.INSTANCE;
}
}
由于InternalSingleton 是私有的,除了 getInstance() 之外没有办法访问它,因此它是懒汉式的;同时读取实例的时候不会进行同步,没有性能缺陷。
4、通过枚举实现懒汉式
通过枚举写单例超级简单,什么都不用想,这也是它最大的优点。下面这段代码就是声明枚举实例的通常做法。
public enum Singleton {
INSTANCE;
public void show(int age) {
System.out.println("this is show function -> " + age);
}
}
我们可以通过Singleton.INSTANCE来访问实例,这比调用getInstance()方法简单多了。创建枚举默认就是线程安全的,所以不需要担心double checked locking,而且还能防止反序列化导致重新创建新的对象。可能由于平时我们使用枚举都是为了表示类别,大家都很少使用这种方式去写单例模式。
关于单例模式的N种实现方式的更多相关文章
- JAVA中单例模式的几种实现方式
1 线程不安全的实现方法 首先介绍java中最基本的单例模式实现方式,我们可以在一些初级的java书中看到.这种实现方法不是线程安全的,所以在项目实践中如果涉及到线程安全就不会使用这种方式.但是如果不 ...
- Python 单例模式的几种实现方式
单例模式的几种实现方式 先来看几个魔法方法的简单运用:__new__, __init__, __call__. class A(object): def __init__(self, x): prin ...
- Python中的单例模式的几种实现方式的优缺点及优化
单例模式 单例模式(Singleton Pattern)是一种常用的软件设计模式,该模式的主要目的是确保某一个类只有一个实例存在.当你希望在整个系统中,某个类只能出现一个实例时,单例对象就能派上用场. ...
- Python中的单例模式的几种实现方式的及优化
单例模式 单例模式(Singleton Pattern)是一种常用的软件设计模式,该模式的主要目的是确保某一个类只有一个实例存在.当你希望在整个系统中,某个类只能出现一个实例时,单例对象就能派上用场. ...
- CSIC_716_20191129【 单例模式 的五种实现方式】
单例模式 单例模式:在确定类中的属性和方法不变时,需要反复调用该类的情况. 让所有通过该类实例化出的对象,都指向同一个内存地址. 优点:节省内存空间. 单例模式有五种表现形式: 1.通过class ...
- 单例模式的两种实现方式对比:DCL (double check idiom)双重检查 和 lazy initialization holder class(静态内部类)
首先这两种方式都是延迟初始化机制,就是当要用到的时候再去初始化. 但是Effective Java书中说过:除非绝对必要,否则就不要这么做. 1. DCL (double checked lockin ...
- C#单例模式的几种实现方式
一.多线程不安全方式实现 public sealed class SingleInstance { private static SingleInstance instance; private S ...
- python 单例模式的四种创建方式
单例模式 单例模式(Singleton Pattern)是一种常用的软件设计模式,该模式的主要目的是确保某一个类只有一个实例存在.当你希望在整个系统中,某个类只能出现一个实例时,单例对象就能派上用场. ...
- Item 3 ------单例模式的几种实现方式,及优缺点
单例模式,是指一个类只有一个唯一的实例,一个类只会被实例化一次.实现这种效果,最佳的方式,编写包含单个元素的枚举类型. 单例模式的最佳实现方式-----创建一个包含单个元素的枚举类 public en ...
随机推荐
- Leetcode 之Evaluate Reverse Polish Notation(41)
很简单的一道题,定义一个栈保留操作数,遇操作符则弹出运算即可. bool isOperator(string &op) { //注意用法 && string("+-* ...
- Codeforces 877C Slava and tanks(思维)
题目链接:http://codeforces.com/problemset 题目大意:有n个格子,某些格子里可能有一个或多个坦克,但不知道具体位置,每个坦克被轰炸一次就会移动到相邻的格子里(第1个格子 ...
- 机器学习方法(四):决策树Decision Tree原理与实现技巧
欢迎转载,转载请注明:本文出自Bin的专栏blog.csdn.net/xbinworld. 技术交流QQ群:433250724,欢迎对算法.技术.应用感兴趣的同学加入. 前面三篇写了线性回归,lass ...
- 760. Find Anagram Mappings
Given two lists Aand B, and B is an anagram of A. B is an anagram of A means B is made by randomizin ...
- img加载不出来,给个默认图片。
忽然发现,jq里也有坑,很多东西莫名其妙的被废弃了……所以,只能用原生js来做了: $('img').each(function() { if (!this.complete || typeof th ...
- 【剑指offer】(第 2 版)Java 题解
[剑指offer](第 2 版)Java 题解 第一章 面试的流程 略... 第二章 面试需要的基础知识 面试题 1. 赋值运算符函数 面试题 2. 实现 Singleton 模式 Solution ...
- SSH Secure File Transfer上传文件错误:encountered 1 errors during the transfer解决办法
在使用SSH 工具向Linux服务器上传文件时,弹出 encountered 1 errors during the transfer 错误. 解决方案: 1.准备上传的那个文件所在目录路径存在(), ...
- ubuntu右上角红色三角警告
问题:ubuntu右上角红色三角感叹号警告,显示更新信息已过期The update information is outdated. 解决方案: 首先在“系统设置(system settings)”- ...
- Python安装scrapy提示 error: Microsoft Visual C++ 14.0 is required. Get it with "Microsoft Visual C++
error: Microsoft Visual C++ 14.0 is required. Get it with "Microsoft Visual C++ Build Tools&quo ...
- Xamarin.Forms教程Android SDK工具下载安装
Xamarin.Form的Android SDK工具下载安装 本节将讲解如何下载Xamarin.Form的Android SDK工具,并使用其中的工具管理Android SDK,如何创建模拟器等内容. ...