单例模式,也叫单子模式,是一种经常使用的软件设计模式。在应用这个模式时,单例对象的类必须保证仅仅有一个实例存在。

很多时候整个系统仅仅须要拥有一个的全局对象。这样有利于我们协调系统总体的行为。比方在某个server程序中,该server的配置信息存放在一个文件中。这些配置数据由一个单例对象统一读取。然后服务进程中的其它对象再通过这个单例对象获取这些配置信息。这样的方式简化了在复杂环境下的配置管理。----维基百科(WIKIPEDIA)

个人的理解:

单例模式概念比較简单,他的目的就是仅仅同意出现一个该类的实例,常常在JDBC操作类等处被用到。我在项目中应用到的地方就是用于获取Dao层的类的实例。单例模式有多种实现方式。这里我认知到的有饿汉式、懒汉式、枚举、静态内部类。庆幸这次整理过程。由于又拓展了不少的单例模式的知识。

以下看实际的样例:

懒汉式



</pre><span style="font-size:18px;"><span style="font-size:12px;"></span></span><pre name="code" class="java">public class Singleton01 {

	private static Singleton01 singleton = null;

	private Singleton01(){}

	public static Singleton01 getInstance(){

		if(singleton == null)
singleton = new Singleton01(); return singleton;
}
}

线程不安全。当多个线程同一时候调用getInstance方法时。都检測到singleton为null,然后就開始创建对象了,这时候就会创建多个实例,而不是一个。

public class Singleton02 {

	private static Singleton02 singleton = null;

	private Singleton02(){}

	public synchronized static Singleton02 getInstance(){

		if(singleton == null)
singleton = new Singleton02(); return singleton;
}
}

这个样例线程安全了,多个线程同一时候调用的时候也不会创建多个对象实例了。可是不是非常高效,须要多个线程依次的去运行这种方法。

public class Singleton03 {

	private static Singleton03 singleton = null;

	private Singleton03(){}

	public static Singleton03 getInstance(){

		if(singleton == null)
synchronized (Singleton03.class) { if(singleton == null){
singleton = new Singleton03();
}
} return singleton;
}
}

这样的方式被称为Double Check(双重检验),比較经典的设计。为什么要两次呢?先说外面的推断是避免对象已经创建好了之后,就不用再去同步运行创建对象去了,第二个是。防止第一次还没有创建实例的时候,多个线程已经在等待中,这个时候须要进行空的推断。防止后面等待的现场进入之后再次创建实例。

參考了其它人的文章【1】:

singleton = new Singleton()不是原子操作,在JVM中大致做了三件事情:

1. 给instance分配内存。

2. 通过其构造函数初始化成员变量。

3. 将instance指向分配的内存空间。

(运行完instance就不是null了)。

可是在JVM编译器中存在指令又一次排序的优化,终于的运行顺序是1-2-3或者1-3-2。假设是1-3-2的话。那么在3运行完成,2运行之前,线程A被线程B抢断的话,instance非空,线程B会直接返回instance,这个时候。直接使用就会报错。

给出的解决方案(使用volatilekeyword屏蔽重排序)是:

public class Singleton03 {

	private volatile static Singleton03 singleton = null;

	private Singleton03(){}

	public static Singleton03 getInstance(){

		if(singleton == null)
synchronized (Singleton03.class) { if(singleton == null){
singleton = new Singleton03();
}
} return singleton;
}
}

饿汉式



public class Singleton04 {

	private static final Singleton04 singleton = new Singleton04();

	private Singleton04(){
} public static Singleton04 getInstance(){
return singleton;
}
}

这样的方式是线程安全的,只是不足的一点是,即使没有调用getInstance方法。也会产生他的实例。之前的懒汉式的方式是懒载入的思想,在须要的时候创建。

静态内部类

public class Singleton05 {

	private static class Singleton{
private static final Singleton05 instance = new Singleton05();
} private Singleton05(){
} public Singleton05 getInstance(){
return Singleton.instance;
}
}

採用了静态内部类的方式,私有的静态内部类,外部无法訪问。类似于饿汉式的形式,可是在此基础上进行了改进,饿汉式是使用的内部属性,而这个是使用的静态内部类的常量。



枚举

神一样的设计,使用枚举实现单例模式的几个原因:

1. 自由序列化

2. 线程安全

3. 保证仅仅有一个实例。

public enum Singleton06 {

	INSTANCE;

	private Singleton06(){
} public void print(){
System.out.println("Enum Singleton!");
}
}

调用:



Singleton06.INSTANCE.print();

代码下载

下载代码

參考

【1】 http://wuchong.me/blog/2014/08/28/how-to-correctly-write-singleton-pattern/



&quot;围观&quot;设计模式(7)--创建型之单例模式(Singleton Pattern)的更多相关文章

  1. 设计模式01 创建型模式 - 单例模式(Singleton Pattern)

    参考 [1] 设计模式之:创建型设计模式(6种) | 博客园 [2] 单例模式的八种写法比较 | 博客园 单例模式(Singleton  Pattern) 确保一个类有且仅有一个实例,并且为客户提供一 ...

  2. 六个创建模式之单例模式(Singleton Pattern)

    定义: 确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例.三个特点:一个类只有一个实例:必需自己创建这个实例:必需自行向整个系统提供这个实例. 结构图: Singleton:单例类,提 ...

  3. 设计模式系列之单例模式(Singleton Pattern)

    单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一.这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式.这种模式涉及到一个单一的类,该类负责创建自己的对象 ...

  4. 设计模式系列之单例模式(Singleton Pattern)——确保对象的唯一性

    模式概述 模式定义 模式结构图 饿汉式单例与懒汉式单例 饿汉式单例 懒汉式单例 模式应用 模式在JDK中的应用 模式在开源项目中的应用 模式总结 主要优点 适用场景 说明:设计模式系列文章是读刘伟所著 ...

  5. 浅谈设计模式--单例模式(Singleton Pattern)

    题外话:好久没写blog,做知识归纳整理了.本来设计模式就是个坑,各种文章也写烂了.不过,不是自己写的东西,缺少点知识的存在感.目前还没做到光看即能记住,得写.所以准备跳入设计模式这个大坑. 开篇先贡 ...

  6. 【设计模式】单例模式 Singleton Pattern

    通常我们在写程序的时候会碰到一个类只允许在整个系统中只存在一个实例(Instance)  的情况, 比如说我们想做一计数器,统计某些接口调用的次数,通常我们的数据库连接也是只期望有一个实例.Windo ...

  7. 设计模式之单例模式(Singleton Pattern)

    单例模式 单例模式(Singleton Pattern)在java中算是最常用的设计模式之一,主要用于控制控制类实例的数量,防止外部实例化或者修改.单例模式在某些场景下可以提高系统运行效率.实现中的主 ...

  8. 乐在其中设计模式(C#) - 单例模式(Singleton Pattern)

    原文:乐在其中设计模式(C#) - 单例模式(Singleton Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 单例模式(Singleton Pattern) 作者:weba ...

  9. Java设计模式之创建型模式

    创建型模式分为五类:工厂方法模式.抽象工厂模式.单例模式.建造者模式.原型模式 一.工厂方法模式:接口-实现类.工厂类

随机推荐

  1. IE 下 input 不响应 change 事件的处理

    很多时候,我们都需要通过 input 来上传文件,通过 change 事件获取用户上传的文件,然后做一些额外的处理,最后上传到服务器. 可是事情往往就是没有那么美好.是的,IE 下 input 在选择 ...

  2. Foundation框架 - NSNumber类

    NSNumber类 NSFormatter #import <Foundation/Foundation.h> int main(int argc, const char * argv[] ...

  3. 绝对让你理解Android中的Context

    这个问题是StackOverFlow上面一个热门的问题What is Context in Android? 整理这篇文章的目的是Context确实是一个非常抽象的东西.我们在项目中随手都会用到它,但 ...

  4. 网络编程——The C10K Problem(C10K = connection 10 kilo 问题)。k 表示 kilo,即 1000

    The C10K problem翻译 (C10K = connection 10 kilo 问题).k 表示 kilo,即 1000 比如:kilometer(千米), kilogram(千克). 如 ...

  5. 【DB2】NOT IN使用中的大坑

    1.环境准备 ------建表TB DROP TABLE TB; CREATE TABLE TB ( ID INTEGER, LEVEL_DETAIL ) ); INSERT INTO TB (ID, ...

  6. PHP-Header缓存策略

    Expires.Cache-Control.Last-Modified.ETag 是RFC 2616(HTTP/1.1)协议中和网页缓存相关的几个字段.前两个用来控制缓存的失效日期,后两个用来验证网页 ...

  7. 解析式/推导式, 生成器 datetime 内建函数

    列表解析式(List Comprehension) 语法: [返回值 for 元素 in 可迭代对象 if 条件] 使用中括号[],内部是for循环,if条件可选. 返回一个新的列表. 列表解析式的作 ...

  8. Silverlight实例教程 - Validation客户端同步数据验证(转载)

    摘要:在Silverlight 4中,Silverlight Validation有相对的改进,本篇将介绍Silverlight 4中新加入的验证机制功能,IDataErrorInfo客户端同步验证机 ...

  9. Atitit. 二进制数据ascii表示法,与base64编码解码api 设计标准化总结java php c#.net

    Atitit. 二进制数据ascii表示法,与base64编码解码api 设计标准化总结java php c#.net 1. Base64编码,1 1.1. 子模式 urlsafe Or  url u ...

  10. iOS开发中邮箱,电话号码,身份证,密码,昵称正则表达式验证

    //邮箱 + (BOOL) validateEmail:(NSString *)email {     NSString *emailRegex = @"[A-Z0-9a-z._%+-]+@ ...