作为程序员这样的特殊物种来说,都掌握了一种特殊能力就是编程思想,逻辑比較慎重,可是有时候总会忽略到一些细节,比方我,一直以来总认为Singleton是设计模式里最简单的,不用太在意,然而就是由于这样的不在意在开发中吃亏了.真的too young to simple.

好不扯淡了,直入主题.

在代码的世界里发现有各种写法的单例,有人说单例有5种,6种,7种…

对于单例的分类这点必须规范下,首先这么多种的分类是依据什么来定义的,基准是什么?

否则怎么会有那么多写法.

因此归纳下来,从延迟载入运行效率的角度出发主要也就分为两种,饿汉顾名思义就是运行效率高,但缺乏延时载入,其它写法差点儿相同都是懒汉式的一个拓展,或者优化而演化出来的,以下请看代码.

开发中经常使用的单例-饿汉式

public class SingletonDemo1 {

    private static final SingletonDemo1 s1 = new SingletonDemo1();

    public static SingletonDemo1 getInstance() {
return s1;
} private SingletonDemo1() {
}
}

没错上面这块代码叫做单例-饿汉式,饿汉式一直以效率高而闻名于单例界,因此咋们开发中经常使用的单例模式也会选择他,简单而好用.

>

开发评价: ★★★★☆

延时载入: ★☆☆☆☆

运行效率: ★★★★★

耗时的蜗牛-懒汉式

public class SingletonDemo2 {
private static SingletonDemo2 s1; public static synchronized SingletonDemo2 getInstance() {
if (s1 == null) {
s1 = new SingletonDemo2();
}
return s1;
} private SingletonDemo2() {
}
}

hello world这个世界里都知道这样的单例基本不会去用,在

double check双重检查锁-懒汉式

这能够说是上面饿汉式的一个缩影,为什么这么说,由于他并不完美,仍然有bug.

public class SingletonDemo3 {
private static SingletonDemo3 s1; public static SingletonDemo3 getInstance() {
if (s1 == null) {
//这里使用了暂时变量
SingletonDemo3 instance;
synchronized (SingletonDemo3.class) {
instance = s1;
if (instance == null) {
synchronized (SingletonDemo3.class) {
if (instance == null) {
instance = new SingletonDemo3();
}
}
s1 = instance;
}
}
}
return s1;
}
}

这个方式主要是通过if推断非null实例,提高了运行的效率,不必每次getInstace都要进行synchronize,仅仅要第一次要同步,有没创建了不用.

可是为什么说这样的写法有bug?这个问题主要是java的jvm层内部模型引起的.简单点说就是instance引用的对象有可能还没有完毕初始化完就直接返回该实例过去,在jdk1.5后这个问题才得到了优化,这不多说,能够看看这篇博文讲得不错.

详情见

当然也有了一些解决方法

  • 使用volatile关键字解决双重检查锁定的bug,对于volatile关键字就是Java中提供的另一种解决可见性和有序性问题的方案.
public class SafeDoubleCheckedLocking {
//加入了volatile关键字
private volatile static Instance instance; public static Instance getInstance() {
if (instance == null) {
synchronized (SafeDoubleCheckedLocking.class) {
if (instance == null)
instance = new Instance();//instance为volatile。如今没问题了
}
}
return instance;
}
}

>

开发评价: ★★★☆☆

延时载入: ★★★☆☆

运行效率: ★★★☆☆

推荐使用的静态内部类-懒汉式

public class SingletonDemo4 {

    //通过静态内部类的方式来实例化对象
private static class InnerSingleton {
private static final SingletonDemo4 instance = new SingletonDemo4();
} public static SingletonDemo4 getInstance() {
return InnerSingleton.instance;
} private SingletonDemo4() {
}
}

这周方式是利用了类载入的一些特性,在classloder的机制中,初始化instance时仅仅有一个线程,并且这样的方式另一个优点就是起到了延时载入的效果,当SingletonDemo4被载入了,可是内部类InnerSingleton并不会被载入,由于InnerSingleton没有主动使用,仅仅有通过调用getInstance方法时才会去载入InnerSingleton类,进而实例private static final SingletonDemo4 instance = new SingletonDemo4();

因此这样的巧妙的方式比起双重检查锁来说,安全来又高效了些.

>

开发评价: ★★★★☆

延时载入: ★★★★☆

运行效率: ★★★★☆

推荐使用的枚举-饿汉式

public enum SingletonDemo5 {

    //枚举元素本身就是一个单例(名字能够任意定义);
INSTANCE; //能够做一些单例的初始化操作
public void singletonOperation() {
}
}

这样的方式事实上非常帅,可是在实际开发中非常少人使用过,这点有点奇怪,首先enum本身就是一个单例,并且枚举另一个特性,能够避免放序列化和反射破解单例问题,经理再也不用操心单例安全了,可能是1.5才有enum的原因吧,假设项目适合的话能够试下这样的单例.

>

开发评价: ★★★★☆

延时载入: ★★★★☆

运行效率: ★★★★☆

总结一下:

对于以下的单例总的来说各有各的优点,至于开发中使用哪个能够依据你的业务需求来选择.

  • 饿汉

    • 标准饿汉 (安全防护方面 枚举单例更优于标准饿汉)

      线程安全,高效,不能够懒载入
    • 枚举单例

      线程安全,高效,不能够懒载入(天然避免反射与反序列化)

  • 懒汉 (效率方面 静态内部类更优于标准懒汉)
    • 标准懒汉

      线程安全,低效,能够懒载入
    • 双重检測(不推荐,有bug)

      线程安全,低效,能够懒载入
    • 静态内部类

      线程安全,低效,能够懒载入

对于单例的安全性问题,能够继续你那帅气的阅读姿势,java中你的单例是不是一直在裸奔

java中你确定用对单例了吗?的更多相关文章

  1. 如何使用双重检查锁定在 Java 中创建线程安全的单例?

    这个 Java 问题也常被问: 什么是线程安全的单例,你怎么创建它.好吧,在Java 5之前的版本, 使用双重检查锁定创建单例 Singleton 时,如果多个线程试图同时创建 Singleton 实 ...

  2. Java中反射和Unsafe破坏单例设计模式

    有如下单例模式设计代码: class Singleton { private String info = "HELLO SHIT"; private static Singleto ...

  3. JAVA之旅(六)——单例设计模式,继承extends,聚集关系,子父类变量关系,super,覆盖

    JAVA之旅(六)--单例设计模式,继承extends,聚集关系,子父类变量关系,super,覆盖 java也越来越深入了,大家加油吧!咱们一步步来 一.单例设计模式 什么是设计模式? JAVA当中有 ...

  4. Unity3D中可中途释放的单例

    Unity3D中可中途释放的单例 使用静态类,静态变量的坏处是从程序加载后就一直占用内存,想要释放比较麻烦,可是之前使用的单例,没有提供释放的方法,那是不是也同静态的一样直到程序结束菜释放?那单例的好 ...

  5. Effective java 第2版 - 笔记(01) 单例(Singleton)的枚举(enum)实现

    直接上代码: public enum Boss { INSTANCE; private String name; public void doSomeThing() { System.out.prin ...

  6. java笔记之静态修饰附和单例设计模式

     第六天笔记 静态修饰符static: 一.static修饰成员变量: static用来修饰成员变量叫静态成员变量,没有static修饰的成员变量叫非静态成员变量 静态成员的访问方式: (1)   用 ...

  7. 【Java】设计模型-五种单例模型

    一. 什么是单例模式 只需要某个类同时保留一个对象,不希望有更多对象,此时,我们则应考虑单例模式的设计. 单例模式的主要作用是保证在Java程序中,某个类只有一个实例存在. 单例模式有很多好处,它能够 ...

  8. Effective Java - 构造器私有、枚举和单例

    目录 饿汉式单例 静态常量 静态代码块 懒汉式单例 尝试加锁 同步代码块 双重检查 静态内部类单例 枚举单例 Singleton 是指仅仅被实例化一次的类.Singleton代表了无状态的对象像是方法 ...

  9. C#中的简单工厂和单例

    下面首先来说说简单工厂 举个例子: 首先是父类 public abstract class Pizza { public abstract string Info(); } } 子类 public c ...

随机推荐

  1. 一款超炫的jquery图片播放插件[Cloud Carousel]

    今天给大家介绍一个jquery图片播放插件,也可以说是一款幻灯片放映插件,它叫Cloud Carousel,支持自动播放.图片预览.鼠标滚轮滚动,非常酷,下图是效果预览. 该jquery图片播放项目演 ...

  2. php过滤文字中的表情字符和mysql服务端对emoji的支持

    1.过滤emoji表情的原因 在我们的项目开发中,emoji表情是个麻烦的东西,即使我们可以能存储,也不一定能完美显示,因为它的更新速度很快:在iOS以外的平台上,例如PC或者android.如果你需 ...

  3. 我为何放弃Gulp与Grunt,转投npm scripts(上)

    本文来源于我在InfoQ中文站翻译的文章.原文地址是:http://www.infoq.com/cn/news/2016/02/gulp-grunt-npm-scripts-part1 Cory Ho ...

  4. [Javascript] Closure Cove, Common mistake

    They’ve got a problem with their existing code, which tries to use a closure. Check it out: function ...

  5. ufldl学习笔记与编程作业:Softmax Regression(vectorization加速)

    ufldl学习笔记与编程作业:Softmax Regression(vectorization加速) ufldl出了新教程,感觉比之前的好.从基础讲起.系统清晰,又有编程实践. 在deep learn ...

  6. Struts2(四)Action一接收参数

    一.属性接收参数并输出 导入struts2的包,导入需要的包 和struts.xml配置文件 <?xml version="1.0" encoding="UTF-8 ...

  7. 简单分页查询(web基础学习笔记十三)

    一.建立资源文件和工具类 1.1 .database.properties jdbc.driver_class=oracle.jdbc.driver.OracleDriver jdbc.connect ...

  8. xcode8 的坑 Info.plist 配置app权限

    好多更新完Xcode8 的小盆友们(我也是小盆友啦),会发现当我们调用系统功能,相册,相机,麦克风等会出现崩溃,而控制台打印出一堆乱七八糟的看不懂的东西,但是最后一句话是有用的,给出了崩溃的原因 啦, ...

  9. JavaScript match()方法使用

    1.JavaScript match() 方法说明http://www.w3school.com.cn/jsref/jsref_match.asp 写法: stringObject.match(sea ...

  10. LeetCode 37 Count and Say

    The count-and-say sequence is the sequence of integers beginning as follows: 1, 11, 21, 1211, 111221 ...