高并发下线程安全的单例模式

1.饿汉式

实现代码:

public class MySingleton {

  // 1.饿汉式

  private static MySingleton singleton = new  MySingleton();

  private MySingleton() {

  }

  public static MySingleton getInstance() {

    return singleton;

  }

}

开启多线程进行测试:

public class SingletonMain extends Thread {

  @Override

  public void run() {

     System.out.println(MySingleton.getInstance().hashCode());

  }

  public static void main(String[] args) {

    // 创建多线程,一起执行看是否生成同一对象

    SingletonMain th1 = new SingletonMain();

    SingletonMain th2 = new SingletonMain();

    SingletonMain th3 = new SingletonMain();

    th1.start();

    th2.start();

    th3.start();

  }

}

结果:

477037219

477037219

477037219

结论:hashCode值一致,说明是同一对象,饿汉式单例模式线程安全

2.懒汉式

实现代码

public class MySingleton {

  // 2.懒汉式

  private static MySingleton instance = null;

  private MySingleton() {

  }

  public static MySingleton getInstance() throws  InterruptedException {

    if (instance == null) {

      // 创建实例之前可能会有一些准备性的耗时工作

      Thread.sleep(300);

      instance = new MySingleton();

    }

    return instance;

  }

}

开启多线程进行测试,

public class SingletonMain extends Thread {

  @Override

  public void run() {

    try {

       System.out.println(MySingleton.getInstance().hashCode());

    } catch (InterruptedException e) {

      e.printStackTrace();

    }

  }

  public static void main(String[] args) {

    // 创建多线程,一起执行看是否生成同一对象

    SingletonMain th1 = new SingletonMain();

    SingletonMain th2 = new SingletonMain();

    SingletonMain th3 = new SingletonMain();

    th1.start();

    th2.start();

    th3.start();

  }

}

结果

1891072390

1139700714

477037219

结论:

不同线程输出的hashCode值不一致,对象不同,懒汉式单例模式线程不安全。

造成不安全的原因:多个线程在进入getInstance方法时,都并没有对象创建出来,导致所有线程都符合(instance == null)的条件,导致对象被重复创建,不构成单例的前提

2-01. 懒汉式改良版-静态同步函数实现

实现代码,增加synchronized关键词,静态同步函数实现线程安全

public class MySingleton {

  // 2.懒汉式

  private static MySingleton instance = null;

  private MySingleton() {

  }

  public static synchronized MySingleton getInstance()  throws InterruptedException {

    if (instance == null) {

      // 创建实例之前可能会有一些准备性的耗时工作

      Thread.sleep(300);

      instance = new MySingleton();

    }

    return instance;

  }

}

结果

1891072390

1891072390

1891072390

结论:

静态同步函数可以实现懒汉式单例模式的线程安全,但是将一整个方法都进行加锁,会导致此方法的运行效率降低,继续考虑其他方式进行改良

2-02.  懒汉式改良版-同步代码块实现

实现代码:增加synchronized关键词,对指定代码块加锁,可以大大的提高整个方法的执行效率,下面使用同步代码块实现

public class MySingleton {

  // 2.懒汉式

  private static MySingleton instance = null;

  private MySingleton() {

  }

  public static MySingleton getInstance() throws  InterruptedException {

    if (instance == null) {

      // 创建实例之前可能会有一些准备性的耗时工作

      Thread.sleep(300);

      synchronized (MySingleton.class) {

        instance = new MySingleton();

      }

    }

    return instance;

  }

}

结果

1891072390

756680587

1994444141

结论:目前此种调整方式还不能达到线程安全,继续改良

双检查锁机制

实现代码

public class MySingleton {

  // 2.懒汉式

  private static MySingleton instance = null;

  private MySingleton() {

  }

  public static MySingleton getInstance() throws  InterruptedException {

    if (instance == null) {

      // 创建实例之前可能会有一些准备性的耗时工作

      Thread.sleep(300);

      synchronized (MySingleton.class) {

        if (instance == null) {

          instance = new MySingleton();

        }

      }

    }

    return instance;

  }

}

结果:

1994444141

1994444141

1994444141

结论:

双检查锁机制可以实现懒汉式单例模式的线程安全,而且这种方式的效率相对于第一种来说是比较高的

3.静态内置类实现单例模式

实现代码

public class MySingleton {

  private static class MySingletonHandler {

    private static MySingleton singleon = new  MySingleton();

  }

  public static MySingleton getInstance() {

    return MySingletonHandler.singleon;

  }

}

开启多线程测试

public class SingletonMain extends Thread {

  @Override

  public void run() {

     System.out.println(MySingleton.getInstance().hashCode());

  }

  public static void main(String[] args) {

    // 创建多线程,一起执行看是否生成同一对象

    SingletonMain th1 = new SingletonMain();

    SingletonMain th2 = new SingletonMain();

    SingletonMain th3 = new SingletonMain();

    th1.start();

    th2.start();

    th3.start();

  }

}

结果

1740545285

1740545285

1740545285

结论:

静态内置类实现单例模式是线程安全的

4.静态代码块实现单例模式

实现代码:静态代码块中的代码在使用类的时候就已经执行了,所以可以应用静态代码块的这个特性的实现单例设计模式。

public class MySingleton {

  // 使用静态代码块实现单例模式

  private static MySingleton singleon = null;

  private MySingleton() {}

  static {

    singleon = new MySingleton();

  }

  public static MySingleton getInstance() {

    return singleon;

  }

}

结果

1209285429

1209285429

1209285429

结论:静态代码块只在类加载时执行,因而只有一个实体对象,从结果来看,线程也是安全的

5.使用枚举数据类型实现单例模式

实现代码

public class ClassFactory {

  private enum MyEnumSingleton {

    singletonFactory;

    private MySingleton instance;

    private MyEnumSingleton() {// 枚举类的构造方法在类加载是被实例化

      instance = new MySingleton();

    }

    public MySingleton getInstance() {

      return instance;

    }

  }

  public static MySingleton getInstance() {

    return MyEnumSingleton.singletonFactory.getInstance();

  }

}

结果

1139700714

1139700714

1139700714

结论:枚举实现单例模式是线程安全的

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,128评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,316评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,737评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,283评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,384评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,458评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,467评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,251评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,688评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,980评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,155评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,818评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,492评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,142评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,382评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,020评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,044评论 2 352