文档章节

Java的几种单例模式及特点

Mrling
 Mrling
发布于 2017/09/08 17:10
字数 1199
阅读 20
收藏 1

1、Java常见的单例模式(懒汉、饿汉、双重锁模式)


       

    1.1饿汉模式   

public class Singleton_e {
	private static final Singleton_e instance = new Singleton_e();
	private Singleton_e() {
		System.out.println("这是单例模式之:饿汉模式");
	}
	public static Singleton_e getInstance() {
		return instance;
	}
}

    从时间和空间方面分析: 饿汉式是典型的空间换时间,当类装载的时候就会创建类实例,不管你用不用,先创建出来,然后每次调用的时候,就不需要再判断了,节省了运行时间。

    从线程安全方面分析:饿汉式是线程安全的,因为虚拟机保证只会装载一次,在装载类的时候是不会发生并发的。

    1.2懒汉模式

public class Singleton_lan {
	private static Singleton_lan instance = null;
	private Singleton_lan() {
		System.err.println("这是单例模式之:懒汉模式");
	}
	public static synchronized Singleton_lan getInstance() {
		if (instance == null) {
			instance = new Singleton_lan();
		}
		return instance;
	}
}

    从时间和空间方面分析: 懒汉式是典型的时间换空间,也就是每次获取实例都会进行判断,看是否需要创建实例,浪费判断的时间。当然,如果一直没有人使用的话,那就不会创建实例,则节约内存空间。
    从线程安全方面分析:懒汉式从线程安全性上讲,不加同步的懒汉式是线程不安全的,比如,有两个线程,一个是线程A,一个是线程B,它们同时调用getInstance方法,那就可能导致并发问题

    1.3双重检查锁模式(DCL)

public class Singleton_shuangchongsuo {
	private static Singleton_shuangchongsuo instance = null;
	private Singleton_shuangchongsuo() {
		System.out.println("这是单例模式之:双重锁模式");//(1)
	}

	public static Singleton_shuangchongsuo getInstance() {
		if (instance == null) {//(2)
			synchronized (instance) {//(3)
				if (instance == null) {//(4)
					instance = new Singleton_shuangchongsuo();//(5)
				}
			}
		}
		return instance;//(6)
	}
}

    双重锁模式简称DCL(double check lock)相当于对懒汉模式进行了增强,保证多线程访问时安全问题(但是不符合happens-before原则,后面会讨论)。

    当然如果直接在Singleton_shuangchongsuo()方法上加锁也可以达到目的,但是这样的话,每次获取实例都要去判断一下是否加锁,造成效率低下。而双重锁只会在特殊情况(第一次创建实例instance时,可能会产生多个实例同时访问的情况下)才会判断锁是哪个,相对效率高。

1.4、用happens-before规则重新审视双重检查锁模式(DCL)

我想简单的用对象创建期间的实际场景来分析一下:(以下是我的个人的理解,所看的资料也是非官方的,不完全保证正确。)

还是看上面1.3双重锁模式的代码,假设线程1执行完(5)时,线程2正好执行到了(2);
看看 new Singleton_shuangchongsuo(); 这个语句的执行过程: 它不是一个原子操作,实际是由多个步骤,我们从我们关注的角度简化一下,简单的认为它主要有2步操作好了:
a) 在内存中分配空间,并将引用指向该内存空间。
b) 执行对象的初始化的逻辑(和操作),完成对象的构建。

此时因为线程1和线程2没有用同步,他们之间不存在“Happens-Before”规则的约束,所以在线程1创建Singleton_shuangchongsuo对象的 a),b)这两个步骤对于线程2来说会有可能出现a)可见,b)不可见
造成了线程2获取到了一个未创建完整的Singleton_shuangchongsuo对象引用,为后边埋下隐患。

改进后的DCL代码:

public class Singleton_lan2 {
	private Singleton_lan2(){
		System.out.println("这是对饿懒模式的的增强,解决由双重锁模式代码未遵循happens-before原则带来的问题");
	}
	private static class InstanceHolder {//1 static 
		private static final  Singleton_lan2 instance = new Singleton_lan2();//2 static 
	}
	public static Singleton_lan2 getInstance(){//3 该static保证不需要实例化Singleton_lan2调用该静态方法
		return InstanceHolder.instance;//通过1、2的static关键字保证可以访问静态类中的静态变量
	}
}

    以上代码是通过静态内部类的方式实现。当getInstance方法第一次被调用的时候,它第一次读取InstanceHolder.instance,导致InstanceHolder类得到初始化;而这个类在装载并被初始化的时候,会初始化它的静态域,从而创建Singleton的实例,由于是静态的域,因此只会在虚拟机装载类的时候初始化一次,并由虚拟机来保证它的线程安全性。这个模式的优势在于,getInstance方法并没有被同步,并且只是执行一个域的访问,因此延迟初始化并没有增加任何访问成本。

 

 

 

更多相关DCL的修正请参考以下文章:

http://www.cs.umd.edu/~pugh/java/memoryModel/DoubleCheckedLocking.html

参考文章:http://ifeve.com/java-concurrent-hashmap-1/

更详细解释可以看这里:http://www.javaeye.com/topic/260515?page=1   

 

 

© 著作权归作者所有

共有 人打赏支持
Mrling
粉丝 2
博文 8
码字总数 7639
作品 0
丰台
程序员
私信 提问
Java单例设计模式的理解与常规实现方式

1:Java中单例模式是一种常见的设计模式,单例模式有以下特点:   单例模式确保某个类只有一个实例,而且自行实例化并向整个系统提供这个实例。 2:java中单例模式的写法也有很多种,我在这...

动力节点
01/02
0
0
设计模式15——Template Method设计模式

Template Method模板方法设计模式定义一个操作中算法的骨架,将具体步骤的执行延迟到子类中实现。Java中的抽象类就是使用了模板方法设计模式。模板方法设计模式结构如下: 以文档处理为例,T...

小米米儿小
2014/01/24
0
0
Java程序员从笨鸟到菜鸟之(三十三)大话设计模式(三)单例模式

本文来自:曹胜欢博客专栏。转载请注明出处:http://blog.csdn.net/csh624366188 单例模式属于对象创建型模式,其意图是保证一个类仅有一个实例,并提供一个访问它的全局访问点。对一些类来说...

长平狐
2012/11/12
120
0
设计模式 2014-12-19

book: 阎宏《JAVA与模式》 架构设计栏目 http://blog.csdn.net/enterprise/column.html 概要: http://bbs.csdn.net/forums/Embeddeddriver 23种设计模式分别是: 1.单例模式 2.工厂方法模式...

jayronwang
2014/12/19
0
0
学了那么多年设计模式依然不会用!那可真蠢!

什么是设计模式? 设计模式(Design Pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。 设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决...

GitChat技术杂谈
2018/10/26
0
0

没有更多内容

加载失败,请刷新页面

加载更多

Jmeter参数的AES加密使用

在Jmeter日常实践中,大家应该都遇到过接口传参需要加密的情况。以登陆为例,用户名和密码一般都需要进行加密传输,在服务端再进行解密,这样安全系数会更高,但在使用jmeter进行接口测试的时...

程序猿拿Q
8分钟前
0
0
MYSQL 日期函数 Date and Time Functions

Table 12.13 Date and Time Functions Name Description ADDDATE() Add time values (intervals) to a date value ADDTIME() Add time CONVERT_TZ() Convert from one time zone to another ......

_liucui_
14分钟前
0
0
Android代码混淆ProGuard工作原理简介

ProGuard能够对Java类中的代码进行压缩(Shrink),优化(Optimize),混淆(Obfuscate),预检(Preveirfy)。    1. 压缩(Shrink): 在压缩处理这一步中,用于检测和删除没有使用的类,字段...

SuShine
17分钟前
0
0
Idea 2018激活

教程地址: https://www.52pojie.cn/thread-781394-1-1.html 亲测可用

一个不正经的程序员
22分钟前
0
0
Android组件化开发实践和案例分享

目录介绍 1.为什么要组件化 1.1 为什么要组件化 1.2 现阶段遇到的问题 2.组件化的概念 2.1 什么是组件化 2.2 区分模块化与组件化 2.3 组件化优势好处 2.4 区分组件化和插件化 2.5 applicatio...

潇湘剑雨
23分钟前
0
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部