`
Thomas_Poppy
  • 浏览: 1270 次
  • 性别: Icon_minigender_1
  • 来自: 成都
社区版块
存档分类
最新评论

单例模式

    博客分类:
  • JAVA
 
阅读更多
1.饿汉单例(Eager)

public class Singleton{

    private static Singleton instance = new Singleton();
    
    private Singleton(){
 
    }
   
    public static Singleton getInstance(){
        return instance;
    }
}


2.懒汉单例(Lazy)


public class Singleton{

    private static Singleton instance=null;
    
    private Singleton(){

    }
   
    public synchronized static Singleton getInstance(){
        if(instance==null){
           instance = new Singleton();
        }
        return instance;
    }
}


3.懒汉单例升级版1(One Checking)

public class Singleton{

    private static Singleton instance=null;
    
    private Singleton(){

    }
   
    public static Singleton getInstance(){
        if(instance==null){       //1 
           synchronized(Singleton.class){ // 2 
              instance = new Singleton(); // 3 
           }
        }
        return instance;
    }
}


备注:
假设两个线程A,B。当线程A执行到1时候,判断instance==null,执行2,这时线程B执行1检测到instance==null,执行2.然后线程A得到类锁,执行3,线程B等待类锁,线程A创建完对象后释放类锁,线程B得到类锁执行3,再次创建对象,系统此时已经创建了两个对象.所以这个升级版有逻辑错误.


4.懒汉单例升级版2(Double Checking)

public class Singleton{

    private static Singleton instance=null;
    
    private Singleton(){

    }
   
    public static Singleton getInstance(){
        if(instance==null){       //1 
           synchronized(Singleton.class){ // 2
              if(instance==null){         // 3
                  instance = new Singleton(); // 4 
              }
           }
        }
        return instance;
    }
}


备注:
这个版本比较好的解决了上面懒汉单例升级版1中出现的问题,当线程B执行到2的时候等类锁,线程A执行完3,4后释放类锁,线程B得到类锁执行3,检测到instance!=null,不执行创建新的对象.这样看来double checking很好地解决了问题.

但是这里存在这样一个问题,当线程A执行到4的时候

按照常理理解应该是执行下面顺序的伪代码:
1.mem=allocate();//Allocate memory for Singleton object.
2.constructSingleton(instance);//Invoke constructor for Singleton 3.passing instance.
instance=mem;//Note that instance is now non-null,but has not been initialized.

但是也可能会这样执行这样顺序的伪代码:
1.mem=allocate();//Allocate memory for Singleton object. (1)
2.instance=mem;//Note that instance is now non-null,but has not been initialized. (2)
3.constructSingleton(instance);//Invoke constructor for Singleton passing instance. (3)

这段伪代码不仅是可能的,而且是一些 JIT 编译器上真实发生的。执行的顺序是颠倒的,

这是因为Java平台内存模型允许所谓的"无序写入"(参考http://blog.csdn.net/hudashi/article/details/6949379),如果是按第二种方式的伪代码执行,那么当线程A执行到4的(2)的时候,假设另外一个线程C执行1,检测到instance!=null,直接返回instance,然而线程A还未执行4的(3),所以线程C这里返回的instance所指向的对象并未初始化,使用的时候会出现问题.


5.懒汉单例升级版3(Double Checking Evolution)

public class Singleton{

    private static Singleton instance=null;
    
    private Singleton(){

    }
   
    public static Singleton getInstance(){
        if(instance==null){
           synchronized(Singleton.class){ 
              Singleton inst=instance;
              if(inst==null){           // 1
                 inst = new Singleton();// 2
              }
              instance=inst; //3
           }
        }
        return instance;
    }
}


备注:
貌似解决了上面的问题,但是 1,2,3处的代码可能会被会被jit compiler优化为
   if(inst==null){
      instance= new Singleton();
   }

优化之后的代码同样存在"无序写入"的问题.


6.懒汉单例升级版4(Inner static class)

public class Singleton {
     private Singleton(){
      
     }

     public static Singleton getInstance(){
         return SingletonInner.instance;
     }

     static class SingletonInner{
         static Singleton instance = new Singleton();
     }
}
分享到:
评论

相关推荐

    43丨单例模式(下):如何设计实现一个集群环境下的分布式单例模式?1

    单例模式是一种设计模式,旨在确保一个类只有一个实例,并提供全局访问点。在单例模式中,类的构造函数是私有的,防止外部直接创建对象,而是通过静态方法获取该类的唯一实例。单例模式的唯一性通常是在进程范围内,...

    设计模式单例模式和工厂模式综合应用

    "设计模式单例模式和工厂模式综合应用"的主题聚焦于两种常用的设计模式:单例模式和工厂模式,并探讨它们如何协同工作来实现高效、灵活的代码结构。这个主题尤其适用于Java编程语言,因为Java的面向对象特性使得设计...

    设计模式之单例模式(结合工厂模式)

    单例模式是软件设计模式中的一种经典模式,它保证了类只有一个实例存在,并提供一个全局访问点。在Java等面向对象编程语言中,单例模式常用于管理共享资源,如数据库连接池、线程池或者配置文件等。结合工厂模式,...

    java单例模式实例

    单例模式是软件设计模式中的一种经典模式,用于确保一个类只有一个实例,并提供一个全局访问点。在Java中,有多种实现单例模式的方法,每种都有其特点和适用场景。接下来,我们将深入探讨这些实现方式。 首先,我们...

    使用C++11实现线程安全的单例模式

    在C++编程中,单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供一个全局访问点。线程安全的单例模式在多线程环境下尤其重要,因为不正确的实现可能导致多个线程创建多个实例,这违反了单例模式...

    设计模式——单例模式

    **设计模式——单例模式** 在软件工程中,设计模式是一种在特定场景下解决常见问题的标准方案,可以被复用并提升代码质量。单例模式是设计模式中的一种,它保证一个类只有一个实例,并提供一个全局访问点。这种模式...

    C#单例模式详解 C#单例模式详解C#单例模式详解

    单例模式是软件设计模式中的一种,它保证一个类只有一个实例,并提供一个全局访问点。在C#中,单例模式常用于管理共享资源或控制类的实例化过程,以提高性能、节约系统资源,特别是在整个应用程序生命周期内只需要一...

    7种单例模式

    单例模式是软件设计模式中的一种经典模式,其主要目的是确保一个类只有一个实例,并提供一个全局访问点。这种模式在很多场景下非常有用,比如控制共享资源、管理配置对象等。下面将详细介绍七种常见的单例模式实现...

    使用单例模式创建学生管理系统(饿汉式、懒汉式)

    单例模式是软件设计模式中的一种,它的主要目的是确保一个类只有一个实例,并提供一个全局访问点。在Java或类似的面向对象编程语言中,单例模式常用于管理共享资源,如数据库连接池、线程池或者配置文件等。在这个...

    c++单例模式线程日志类

    在C++编程中,单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供一个全局访问点。在这个特定的场景中,我们讨论的是一个实现了单例模式的日志类,该类专为多线程环境设计,具备日志等级控制、...

    单例模式实现mdi界面子窗体控制

    首先向关注过我这个系列...这立刻让我想到了最常用也是最简单最容易理解的一个设计模式 单例模式 何为 单例模式 ? 故名思议 即 让 类 永远都只能有一个实例。 由于 示例代码 比较简单 我也加了注释,这里就不在赘述

    几种单例模式demo

    单例模式是软件设计模式中的一种,它的主要目的是确保一个类只有一个实例,并提供一个全局访问点。这种模式在很多场景下都非常有用,比如控制资源的唯一性、管理共享配置或者创建昂贵的对象时避免频繁创建销毁。 ...

    Java 单例模式 工具类

    Java中的单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供全局访问点。在Java编程中,单例模式常用于控制资源的访问,比如数据库连接池、线程池或者日志对象等。本篇文章将深入探讨如何在Java中...

    单例模式详解~~单例模式详解~~

    单例模式是一种设计模式,它的主要目标是确保一个类只有一个实例,并提供一个全局访问点。在软件工程中,单例模式常用于控制资源的共享,比如数据库连接池、线程池或者日志系统等,这些资源通常需要全局唯一且高效地...

    单例模式(singleton)

    单例模式是软件设计模式中的一种,它的核心思想是确保一个类在整个系统中只有一个实例,并提供一个全局访问点。在Java或类似编程语言中,单例模式常常被用来管理资源,比如数据库连接、线程池或者配置信息,因为这些...

    Qt qml Singleton 单例模式

    在Qt的Qml环境中,单例模式是一种设计模式,它允许在整个应用程序中创建一个全局访问点,确保某个类只有一个实例存在。这样的设计模式在需要共享数据或者服务时非常有用,避免了多处创建相同对象导致的数据不一致或...

    设计模式之单例模式源码demo

    单例模式是软件设计模式中的经典模式之一,其主要目的是控制类的实例化过程,确保在应用程序的整个生命周期中,某个类只有一个实例存在。这样的设计通常适用于那些需要频繁创建和销毁,但资源消耗较大的对象,如...

    23钟设计模式之单例模式

    单例模式是一种常用的设计模式,它的核心思想是在整个应用程序中,一个类只能有一个实例存在。单例模式常用于控制资源的共享,例如数据库连接池、日志服务等。单例模式有多种实现方式,常见的包括懒汉式、饿汉式以及...

    设计模式C++学习之单例模式(Singleton)

    单例模式是软件设计模式中的一种,它保证一个类只有一个实例,并提供一个全局访问点。在C++中,实现单例模式有多种方法,我们将会深入探讨这一模式的原理、优缺点以及如何在实际编程中应用。 单例模式的核心在于...

Global site tag (gtag.js) - Google Analytics