饿汉式(推荐)

  1. package concurencyv2.chapter1;
  2. public class SingletonV2 {
  3. private static final SingletonV2 instance = new SingletonV2();
  4. private SingletonV2() {}
  5. public static SingletonV2 getInstance() {
  6. return instance;
  7. }
  8. }

优点:初试化静态的instance创建一次。如果我们在Singleton类里面写一个静态的方法不需要创建实例,它仍然会早早的创建一次实例。而降低内存的使用率。

缺点:没有lazy loading的效果,从而降低内存的使用率。

单线程下

  1. package concurencyv2.chapter1;
  2. public class SingletonV1 {
  3. private static SingletonV1 instance = null;
  4. private SingletonV1() {}
  5. public SingletonV1 getInstance() {
  6. if(null == instance)
  7. instance = new SingletonV1();
  8. return SingletonV1.instance;
  9. }
  10. }

注解: Singleton的静态属性instance中,只有instance为null的时候才创建一个实例,构造函数私有,确保每次都只创建一个,避免重复创建。

缺点:4只在单线程的情况下正常运行,在多线程的情况下,就会出问题。例如:当两个线程同时运行到判断instance是否为空的if语句,并且instance确实没有创建好时,那么两个线程都会创建一个实例。

懒汉式

  1. package concurencyv2.chapter1;
  2. public class SingletonV3 {
  3. private SingletonV3() {
  4. }
  5. private static SingletonV3 instance;
  6. public synchronized static SingletonV3 getInstance() {
  7. if(null == instance)
  8. instance = new SingletonV3();
  9. return SingletonV3.instance;
  10. }
  11. }

注解:在单线程的基础上加上了同步锁,使得在多线程的情况下可以用。例如:当两个线程同时想创建实例,由于在一个时刻只有一个线程能得到同步锁,当第一个线程加上锁以后,第二个线程只能等待。第一个线程发现实例没有创建,创建之。第一个线程释放同步锁,第二个线程才可以加上同步锁,执行下面的代码。由于第一个线程已经创建了实例,所以第二个线程不需要创建实例。保证在多线程的环境下也只有一个实例。

缺点:每次通过getInstance方法得到singleton实例的时候都有一个试图去获取同步锁的过程。而众所周知,加锁是很耗时的。能避免则避免。

double check

  1. package concurencyv2.chapter1;
  2. public class SingletonV4 {
  3. private SingletonV4() {
  4. }
  5. private static SingletonV4 instance;
  6. public static SingletonV4 getInstance() {
  7. if(null == instance) {
  8. synchronized (SingletonV4.class) {
  9. if(null == instance)
  10. instance = new SingletonV4();
  11. }
  12. }
  13. return SingletonV4.instance;
  14. }
  15. }

注解:只有当instance为null时,需要获取同步锁,创建一次实例。当实例被创建,则无需试图加锁。

缺点: 可能会出现空指针异常,一个线程获取了同步锁,并且创建了,但是还没有完成初始化。 另外一个线程直接getInstace,因此这个线程可能获取到的对象,有些地方没有初始化完成,造成引用的空指针现象。

double check and add volatile (推荐)

  1. package concurencyv2.chapter1;
  2. public class SingletonV5 {
  3. private SingletonV5() {
  4. }
  5. private static volatile SingletonV5 instance;
  6. public static SingletonV5 getInstance() {
  7. if(null == instance) {
  8. synchronized (SingletonV5.class) {
  9. if(null == instance)
  10. instance = new SingletonV5();
  11. }
  12. }
  13. return SingletonV5.instance;
  14. }
  15. }

优点:在instance上添加了volatile,使得每次执行读操作的时候保证写操作已经完成.

静态内部类 (推荐)

  1. package concurencyv2.chapter1;
  2. public class SingletonV6 {
  3. private SingletonV6() {}
  4. private static class SingletonHolder {
  5. public static final SingletonV6 instance = new SingletonV6();
  6. }
  7. public SingletonV6 getInstance() {
  8. return SingletonHolder.instance;
  9. }
  10. }

枚举enum

  1. package concurencyv2.chapter1;
  2. public class SingletonV7 {
  3. private SingletonV7() {
  4. }
  5. private enum Singleton {
  6. SINGLETON;
  7. private SingletonV7 instance;
  8. Singleton() {
  9. instance = new SingletonV7();
  10. }
  11. }
  12. public static SingletonV7 getInstance() {
  13. return Singleton.SINGLETON.instance;
  14. }
  15. }

利用enum只初始化一次的特性,保证了线程安全性.

快速理解Java中的七种单例模式的更多相关文章

  1. 快速理解Java中的五种单例模式

    解法一:只适合单线程环境(不好) package test; /** * @author xiaoping * */ public class Singleton { private static S ...

  2. 快速理解Java中的五种单例模式(转)

    解法一:只适合单线程环境(不好) package test; /** * @author xiaoping * */ public class Singleton { private static S ...

  3. Java中的五种单例模式实现方法

    [代码] Java中的五种单例模式实现方法   1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 2 ...

  4. java中的几种单例模式

    目前比较常见的有4种(DCL为懒汉模式的线程安全版本). 单例模式的实现一般需要满足以下条件: 1.构造方法私有化,实例属性私有化. 2.必须仅在类的内部完成实例的初始化过程. 3.提供公共静态方法, ...

  5. 5000字 | 24张图带你彻底理解Java中的21种锁

    本篇主要内容如下: 本篇文章已收纳到我的Java在线文档. Github 我的SpringCloud实战项目持续更新中 帮你总结好的锁: 序号 锁名称 应用 1 乐观锁 CAS 2 悲观锁 synch ...

  6. Java中的五种单例模式

    Java模式之单例模式: 单例模式确保一个类只有一个实例,自行提供这个实例并向整个系统提供这个实例. 特点: 1,一个类只能有一个实例 2 自己创建这个实例 3 整个系统都要使用这个实例 例: 在下面 ...

  7. JDK学习---深入理解java中的HashMap、HashSet底层实现

    本文参考资料: 1.<大话数据结构> 2.http://www.cnblogs.com/dassmeta/p/5338955.html 3.http://www.cnblogs.com/d ...

  8. 理解Java中的弱引用(Weak Reference)

    本篇文章尝试从What.Why.How这三个角度来探索Java中的弱引用,理解Java中弱引用的定义.基本使用场景和使用方法.由于个人水平有限,叙述中难免存在不准确或是不清晰的地方,希望大家可以指出, ...

  9. JAVA基础学习之throws和throw的区别、Java中的四种权限、多线程的使用等(2)

    1.throws和throw的区别 throws使用在函数外,是编译时的异常,throw使用在函数内,是运行时的异常 使用方法 public int method(int[] arr) throws ...

随机推荐

  1. SV通过DPI调用C

    Verilog与C之间进行程序交互,PLI(Programming Language Interface)经过了TF,ACC,VPI等模式. 使用PLI可以生成延时计算器,来连接和同步多个仿真器,并可 ...

  2. 六 js函数和this

    js的所有代码都是由funtion组成,funtion即函数的类型. 一.函数有两种写法 -----1.定义式 function test() { //定义一个函数 console.log(" ...

  3. flask上下文全局变量,程序上下文、请求上下文、上下文钩子

    Flask上下文 Flask中有两种上下文,程序上下文(application context)和请求上下文(request context) 当客户端发来请求时,请求上下文就登场了.请求上下文里包含 ...

  4. C# 声明隐式类型的局部变量

    在c#中赋值给变量的值必须具有和变量相同的类型.如int值赋给int变量,c#编译器可以迅速判断变量初始化表达式的类型,如果变量类型不符,就会明确告诉你. 提示需要强制转换(例如在char中不允许使用 ...

  5. HDU 3172 Virtual Friends (map+并查集)

    These days, you can do all sorts of things online. For example, you can use various websites to make ...

  6. AtCoder Regular Contest 077 C - pushpush

    题目链接:http://arc077.contest.atcoder.jp/tasks/arc077_a Time limit : 2sec / Memory limit : 256MB Score ...

  7. Symfony2 学习笔记之控制器

    一个controller是你创建的一个PHP函数,它接收HTTP请求(request)并创建和返回一个HTTP回复(Response).回复对象(Response)可以是一个HTML页面,一个XML文 ...

  8. Oracle之表的相关操作

    #添加字段 格式: alter table table_name add column_name datatype; 例子: alter table userinfo ); desc userinfo ...

  9. v-text v-html等指令的使用

    v-text:以纯文本方式显示数据: v-html:可以识别HTML标签: v-once:只渲染元素或组件一次: v-pre:不进行编译,直接显示内容: v-cloak:可以隐藏未编译的 Mustac ...

  10. python之接口与抽象类

    一.接口与归一化设计 1.什么是接口 1)是一组功能集合 2)接口的功能是用于交互 3)接口只定义函数,但不涉及函数的实现 4)这些功能是相关的 2.为什么要用接口 接口提取了一群类共同的函数,然后让 ...