• 首页 首页 icon
  • 工具库 工具库 icon
    • IP查询 IP查询 icon
  • 内容库 内容库 icon
    • 快讯库 快讯库 icon
    • 精品库 精品库 icon
    • 问答库 问答库 icon
  • 更多 更多 icon
    • 服务条款 服务条款 icon

单例模式饿汉实现、懒汉实现

武飞扬头像
Youcan.
帮助1

目录

(MS)

1. 饿汉实现 

2.  懒汉模式

① 单线程版本

② 线程安全版

③ 线程安全改进版⼀

以上存在线程安全问题。

④ 线程安全改进版⼆

以上存在线程安全问题,对象创建需要三步:

⑤ 线程安全改进最终版


(MS)

学新通

单例模式具体的实现⽅式, 分成 "饿汉" 和 "懒汉" 两种:
● 饿汉⽅式:程序启动之后,里面创建单例对象。(线程安全,可能会造成资源浪费)
● 懒汉⽅式:当有程序调用单例对象的时候才初始化。(使⽤时才加载,可以避免资源不必要的浪费)

单例模式的实现:

无论是饿汉还是懒汉,实现的模式是一模一样的,分为3种:

        1. 创建 一个私有的构造函数(为防止其他类直接new此对象)。

        2. 创建一个私有的属性对象。

        3. 创建一个公共的对外暴露的单例对象。

默认的懒汉模式是⾮线程安全的,使⽤要对懒汉进⾏优化,优化改进:
        ● 全局加锁(线程安全、性能⽐较低)
        ● 局部加锁(线程安全,双重效验)

1. 饿汉实现 

程序执行起来之后,执行是一次性的,是线程安全的,会造成资源的浪费。

  1.  
    饿汉实现
  2.  
    public class DataSourceSingleton {
  3.  
    //1. 提供私有的构造方法,(防止外部直接new对象)
  4.  
    private DataSourceSingleton() {
  5.  
    }
  6.  
     
  7.  
    //2. 创建一个私有的属性对象
  8.  
    private static DataSourceSingleton dataSource = new DataSourceSingleton();
  9.  
     
  10.  
    //提供公共的对外的单例对象
  11.  
    public static DataSourceSingleton getInstance() {
  12.  
    return dataSource;
  13.  
    }
  14.  
    }

2.  懒汉模式

为了解决饿汉模式造成的资源浪费,产生了懒汉模式。

① 单线程版本

  1.  
    单例模式——懒汉模式 1 单线程版本
  2.  
     
  3.  
    public class DataSourceSingleton2 {
  4.  
    // 1.私有的构造方法
  5.  
    private DataSourceSingleton2() {
  6.  
    }
  7.  
     
  8.  
    // 2.创建一个私有的属性
  9.  
    private static DataSourceSingleton2 dataSource = null;
  10.  
     
  11.  
    // 3.创建一个对外提供访问的单例对象
  12.  
    public static DataSourceSingleton2 getInstance() {
  13.  
    if (dataSource == null) { // 第一次访问
  14.  
    dataSource = new DataSourceSingleton2();
  15.  
    }
  16.  
    return dataSource;
  17.  
    }
  18.  
    }
学新通

 此懒汉模式的实现是线程不安全的.

线程安全问题发⽣在⾸次创建实例时. 如果在多个线程中同时调⽤ getInstance ⽅法, 就可能导致创建出多个实例.
⼀旦实例已经创建好了, 后⾯再多线程环境调⽤ getInstance 就不再有线程安全问题了(不再修改 instance 了)。

② 线程安全版

  1.  
    单例模式——懒汉模式2 线程安全版
  2.  
    public class DataSourceSingleton3 {
  3.  
    // 1.私有的构造方法
  4.  
    private DataSourceSingleton3() {
  5.  
    }
  6.  
     
  7.  
    // 2.创建一个私有的属性
  8.  
    private static DataSourceSingleton3 dataSource;
  9.  
     
  10.  
    // 3.创建一个对外提供访问的单例对象
  11.  
    public synchronized static DataSourceSingleton3 getInstance() {
  12.  
    if (dataSource == null) { // 第一次访问
  13.  
    //可能会出问题
  14.  
    dataSource = new DataSourceSingleton3();
  15.  
    }
  16.  
    return dataSource;
  17.  
    }
  18.  
    }
学新通

虽然线程安全,但给全局加锁,性能不好。  

③ 线程安全改进版⼀

  1.  
    单例模式--懒汉模式3 线程安全改进版⼀
  2.  
    public class DataSourceSingleton4 {
  3.  
    // 1. 私有的构造方法
  4.  
    private DataSourceSingleton4() {
  5.  
     
  6.  
    }
  7.  
    // 2. 私有的属性
  8.  
    private static DataSourceSingleton4 dataSource = null;
  9.  
    // 3. 公共的访问方法,得到单例对象
  10.  
    public static DataSourceSingleton4 getInstance() {
  11.  
    //在返回dataSourse的之前,要先判断,是不是第一次访问,如果是第一次访问,需要new一个对象,此时,存在线程安全问题
  12.  
     
  13.  
    if(dataSource == null) { // 大致分流状态
  14.  
    synchronized (DataSourceSingleton4.class) {
  15.  
    dataSource = new DataSourceSingleton4(); // 加上synchronized (),问题仍没有解决
  16.  
    }
  17.  
    }
  18.  
    return dataSource;
  19.  
    }
  20.  
    }
学新通

以上存在线程安全问题。

④ 线程安全改进版⼆

  1.  
    单例模式--懒汉模式4 线程安全改进版⼆
  2.  
    public class DataSourceSingleton5 {
  3.  
    // 1. 私有的构造方法
  4.  
    private DataSourceSingleton5() {
  5.  
    }
  6.  
    // 2. 私有的属性
  7.  
    private static DataSourceSingleton5 dataSource = null;
  8.  
     
  9.  
    // 3. 公共的访问方法,得到单例对象
  10.  
    public static DataSourceSingleton5 getInstance() {
  11.  
    //在返回dataSourse的之前,要先判断,是不是第一次访问,如果是第一次访问,需要new一个对象,此时,存在线程安全问题
  12.  
    // 加上synchronized (),问题仍没有解决
  13.  
    //在synchronized ()内部再次判断是不是第一次访问
  14.  
    if(dataSource == null) { // 大致分流状态
  15.  
    synchronized (DataSourceSingleton4.class) {
  16.  
    if(dataSource == null) { // 精细化的分流(DCL双重效验锁)
  17.  
    dataSource = new DataSourceSingleton5();
  18.  
    }
  19.  
    }
  20.  
    }
  21.  
    return dataSource;
  22.  
    }
  23.  
    }
学新通

以上存在线程安全问题,对象创建需要三步:

        1. 给对象分配内存

        2. 初始化对象

        3. 设置对象到相应的内存地址上

假如因为指令重排导致执⾏的顺序变为了132,那么假如a线程中执⾏完1、3之后,b线程到达代码2处.执⾏判断语句,发现instance指向的是⼀段地址,因此直接不进⼊判断语句⽽是直接返回了⼀个没有初始化的空的对象。

⑤ 线程安全改进最终版

  1.  
    单例模式--懒汉模式5 线程安全改进最终版
  2.  
    public class DataSourceSingleton6 {
  3.  
    // 1. 私有的构造方法
  4.  
    private DataSourceSingleton6() {
  5.  
     
  6.  
    }
  7.  
    // 2. 私有的属性
  8.  
    private static volatile DataSourceSingleton6 dataSource = null;
  9.  
     
  10.  
    // 3. 公共的访问方法,得到单例对象
  11.  
    public static DataSourceSingleton6 getInstance() {
  12.  
    //在返回dataSourse的之前,要先判断,是不是第一次访问,如果是第一次访问,需要new一个对象,此时,存在线程安全问题
  13.  
    // 加上synchronized (),问题仍没有解决
  14.  
    //在synchronized ()内部再次判断是不是第一次访问
  15.  
    if(dataSource == null) { // 大致分流状态
  16.  
    synchronized (DataSourceSingleton4.class) {
  17.  
    if(dataSource == null) { // 精细化的分流(DCL双重效验锁)
  18.  
    dataSource = new DataSourceSingleton6();
  19.  
    }
  20.  
    }
  21.  
    }
  22.  
    return dataSource;
  23.  
    }
  24.  
    }
学新通

学新通

这篇好文章是转载于:学新通技术网

  • 版权申明: 本站部分内容来自互联网,仅供学习及演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,请提供相关证据及您的身份证明,我们将在收到邮件后48小时内删除。
  • 本站站名: 学新通技术网
  • 本文地址: /boutique/detail/tanhgcegcg
系列文章
更多 icon
同类精品
更多 icon
继续加载