《Java与模式》学习笔记——Singleton

Posted brooksychen

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了《Java与模式》学习笔记——Singleton相关的知识,希望对你有一定的参考价值。

单例(Singleton)模式的要点有三个:一是某个类只能有一个实例;二是它必须自行创建这个实例;三是它必须自行向整个系统提供这个实例。一些资源管理器常常设计成单例模式。   单例模式的结构如下:   饿汉式单例类: public   class  EagerSingleton 
    
private EagerSingleton() 
    
public static EagerSingleton getInstance() 
        
return m_instance;
    

    
private static final EagerSingleton m_instance = new EagerSingleton();

  由于构造函数是私有的,因此此类不能被继承。   懒汉式单例类: public   class  LazySingleton 
    
private LazySingleton() 
    
synchronized public static LazySingleton getInstance() 
        
if (m_instance == null
            m_instance 
= new LazySingleton();
        

        
return m_instance;
    

    
private static LazySingleton m_instance = null;

  上例的静态工厂方法使用了同步化,以处理多线程环境。   饿汉式单例类在自己被加载时就将自己实例化,即使加载器是静态的,在饿汉式单例类被加载时仍会将自己实例化。单从资源利用效率角度来讲,这个比懒汉式单例类稍差些。从速度和反应时间角度来讲,则比懒汉式单例类稍好些。然而,懒汉式单例类在实例化时,必须处理好在多个线程同时首次引用此类时的访问限制问题,特别是当单例类作为资源控制器在实例化时必须涉及资源初始化,而资源初始化很有可能耗费时间。这意味着出现多线程同时首次引用此类的几率变得较大。   饿汉式单例类可以在Java语言内实现,但不易在C++内实现,因为静态初始化在C++里没有固定的顺序,因而静态的m_instance变量的初始化与类的加载顺序没有保证,可能会出问题。   登记式单例类: 登记式单例类是GoF为了克服饿汉式单例类及懒汉式单例类均不可继承的缺点而设计的。它的实例化方式可以是饿汉式或懒汉式的,但它的子类实例化的方式只能是懒汉式的。 package  com.javapatterns.singleton.demos;

import  java.util.HashMap;

public   class  RegSingleton 

    
protected RegSingleton() 

    
static public RegSingleton getInstance(String name) 
        
if (name == null
            name 
= "com.javapatterns.singleton.demos.RegSingleton";
        

        System.out.println(
"From RegSingleton: requesting for " + name );
        
if (m_registry.get(name) == null
            
try 
                m_registry.put( name, Class.forName(name).newInstance() ) ;
            
 catch(ClassNotFoundException e) 
                System.out.println(
"Class " + name + " is not found.");
            
 catch(InstantiationException e) 
                System.out.println(
"Class " + name + " can not be instantiated.");
            
 catch(IllegalAccessException e) 
                System.out.println(
"Class " + name + " can not be accessed.");
            

        

        
return  (RegSingleton) (m_registry.get(name));
    


    
static private HashMap m_registry = new HashMap();

    
static 
        RegSingleton x 
= new RegSingleton();
        m_registry.put( x.getClass().getName() , x);
    


    
public String about() 
        
return "Hello, I am RegSingleton.";
    


  package  com.javapatterns.singleton.demos;

import  java.util.HashMap;

public   class  RegSingletonChild  extends  RegSingleton 

    
public RegSingletonChild() 

    
static public RegSingletonChild getInstance() 
        
return (RegSingletonChild) RegSingleton.getInstance( "com.javapatterns.singleton.demos.RegSingletonChild" );
    


    
public String about() 
        
return "Hello, I am RegSingletonChild.";
    


  由于子类必须允许父类以构造函数调用产生实例,因此,它的构造函数必须是公开的。这样一来,就等于允许了以这样的方式产生实例而不在父类的登记中。这是登记式单例类的一个缺点。由于父类的实例必须存在才可能有子类的实例,这在有些情况下是一个浪费。这是登记式单例类的另一个缺点。   一个单例类可以是有状态的(stateful),一个有状态的单例对象一般也是可变(mutable)单例对象。单例类也可以是没有状态的(stateless),一个没有状态的单例类也就是不变(immutable)单例类。   在任何使用了EJB、RMI和JINI技术的分散式系统中,应当避免使用有状态的单例模式。   除非系统有协调机制,不然在有多个类加载器的情况下应当尽量避免使用有状态的单例类。   Java语言中,像Runtime、java.awt.Toolkit类都是单例类。   默认实例模式(Default Instance Pattern): 有些设计师将不完全的单例模式叫做“默认实例模式”。在所谓的“默认实例模式”里面,一个类提供静态的方法,如同单例模式一样,同时又提供一个公开的构造函数,如同普通的类一样。这样做的唯一好处是,这种模式允许客户端选择如何将类实例化:创建新的自己独有的实例,或者使用共享的实例。

以上是关于《Java与模式》学习笔记——Singleton的主要内容,如果未能解决你的问题,请参考以下文章

《Java与模式》学习笔记——Singleton

Java学习笔记(二十四):单例设计模式singleton

Java:Effective java学习笔记之 用私有构造器或者枚举类型强化SIngleton属性

一文彻底搞懂单例模式(Singleton-Pattern)

单例模式的学习笔记

java设计模式学习 ----- 单例模式(Singleton)