`
uule
  • 浏览: 6348626 次
  • 性别: Icon_minigender_1
  • 来自: 一片神奇的土地
社区版块
存档分类
最新评论

单例模式

 
阅读更多

Singleton模式

主要作用是保证一个类Class只有一个实例存在。

注意默认构造函数。

注意DCL方式中变量声明为volatile

几种形式:

饿汉:

这种方式基于classloder机制避免了多线程的同步问题,不过,instance在类装载时就实例化,虽然导致类装载的原因有很多种,在单例模式中大多数都是调用getInstance方法, 但是也不能确定有其他的方式(或者其他的静态方法)导致类装载,这时候初始化instance显然没有达到lazy loading的效果。

Java代码 
  1. public class Singleton {  
  2.   
  3.   private Singleton(){}  
  4.   
  5.   //在自己内部定义自己一个实例,是不是很奇怪?  
  6.   //注意这是private 只供内部调用  
  7.   
  8.   private static Singleton instance = new Singleton();  
  9.   
  10.   //这里提供了一个供外部访问本class的静态方法,可以直接访问    
  11.   public static Singleton getInstance() {  
  12.     return instance;     
  13.    }  
  14. }   

 饿汉变种:

 

Java代码 
  1. public class Singleton {    
  2.     private Singleton instance = null;    
  3.       
  4.     static {    
  5.         instance = new Singleton();    
  6.     }   
  7.        
  8.     private Singleton (){}  
  9.           
  10.     public static Singleton getInstance() {    
  11.         return this.instance;    
  12.     }    
  13.  }  

 

 

 

第二种形式:

懒汉:

效率很低。

Java代码 
  1. public class Singleton {  
  2.   
  3.   private static Singleton instance = null;  
  4.   
  5.   public static synchronized Singleton getInstance() {  
  6.   
  7.   if (instance==null)  
  8.     instance=new Singleton();  
  9.   return instance;   }  
  10.   
  11. }   

第三种:

静态内部类:

这种方式同样利用了classloder的机制来保证初始化instance时只有一个线程,它跟饿汉不同的是(很细微的差别):饿汉方式是只要Singleton类被装载了,那么instance就会被实例化(没有达到lazy loading效果),而这种方式是Singleton类被装载了,instance不一定被初始化。因为SingletonHolder类没有被主动使用,只有显式通过调用getInstance方法时,才会显示装载SingletonHolder类,从而实例化instance。

 

静态内部类在外部类加载时不会初始化么?

 

内部类也是个类啊,和外部类一样,什么时候会触发该类的初始化构成呢?当然是对该类的变量进行使用,或者是调用了针对该类的Class.forName()方法

Java代码 
  1. public class Singleton {    
  2.     
  3.     private static class Holder {    
  4.         private static final Singleton instance = new Singleton();    
  5.     }    
  6.     
  7.     private Singleton() {    
  8.     }    
  9.     
  10.     public static Singleton getInstance() {    
  11.         return Holder.instance;    
  12.     }    
  13. }    

 

上面第二中形式是lazy initialization,也就是说第一次调用时初始Singleton,以后就不用再生成了。

注意到lazy initialization形式中的synchronized,这个synchronized很重要,如果没有synchronized,那么使用getInstance()是有可能得到多个Singleton实例。关于lazy initialization的Singleton有很多涉及double-checked locking (DCL)的讨论,有兴趣者进一步研究。

一般认为第一种形式要更加安全些。

 

 

Java代码 
  1. //DCL  
  2. public class LazySingleton {       
  3.     private static volatile LazySingleton instance;       
  4.            
  5.     public static LazySingleton getInstantce() {       
  6.         if (instance == null) {       
  7.             synchronized (LazySingleton.class) {       
  8.                 if (instance == null) {       
  9.                     instance = new LazySingleton();       
  10.                 }    
  11.             }       
  12.         }       
  13.         return instance;       
  14.     }       
  15. }   

 当然还有一种方式是使用重入锁ReentrantLock.

 

枚举(enum)方式

Java代码 
  1. public enum Singleton {  
  2.   instance;  
  3. }  

这种方式是《Effective Java》作者Josh Bloch 提倡的方式

优点:能避免多线程同步问题,而且还能防止反序列化重新创建新的对象

缺点:这种方式是有限制的,你如果enum用得多就知道了,enum有很多限制,毕竟不是普通的类。

还有就是enum的单例的效果是等价于饿汉式初始化方式的,并不是最理想的需要单例时才执行单例初始化代码的模式。

 

enum自身有限制的么?例如enum不能继承

 

很多时候为了减少类的数量,或者某个单例类中有几个public static的公共方法,或者是有几个public static的常量,此时用我推荐的方式实现单例就是最完美的。

 

完美的定义:

1.加载这个类的时候不会提引发单例的初始化。

2.即使调用该类的public static方法或public static的常量也不会引提前引发单例的初始化。

3.只有在需要的时候才会初始化单例,而且不需要加同步控制,由JDK自身的classloader机制来完成高效的同步控制,这里的高效是值,只有第一次初始化的时候可能产生竞争,一旦初始化完毕不再产生同步竞争。

 

enum方式的单例只是符合上述第三条而已。碰到混合型功能的类,或者当前单例需要继承其他类,或者需要有Spring来管理一些有状态的bean的注入的话,enum就不能满足要求了。

 

最佳是指: 

1.运行效率大大优于懒汉式,因为懒汉式有一个同步过程(因为Java中无法像C++那样用双重检查,所以那个同步无法避开)。 

2.同时又不像饿汉式那样,过早的触发初始化。这个过早是指: 

如果有如下的变量: 

public static int totalNum = 0; 

如果是恶汉式的话,一调用这个变量,这个类就会初始化,而用类加载的方式的话,不会这么早就开始单例初始化。

 

Effective java Second Edition 第二章第3条:用私有构造器或枚举类型强化Singleton属性。里面有各种单例模式的详细比较。

 

 

总结

有两个问题需要注意:

     1、如果单例由不同的类装载器装入,那便有可能存在多个单例类的实例。假定不是远端存取,例如一些servlet容器对每个servlet使用完全不同的类  装载器,这样的话如果有两个servlet访问一个单例类,它们就都会有各自的实例。

     2、如果Singleton实现了java.io.Serializable接口,那么这个类的实例就可能被序列化和复原。不管怎样,如果你序列化一个单例类的对象,接下来复原多个那个对象,那你就会有多个单例类的实例。

 

对第一个问题修复的办法是:

 

private static Class getClass(String classname)      
                               throws ClassNotFoundException {     
							   
       ClassLoader classLoader = Thread.currentThread().getContextClassLoader();     
       
       if(classLoader == null)     
          classLoader = Singleton.class.getClassLoader();     
       
       return (classLoader.loadClass(classname));     
    }     
 }  

  

 对第二个问题修复的办法是: 

public final class MySingleton implements Serializable{
    private MySingleton() { }
	
    private static final MySingleton INSTANCE = new MySingleton();
	
    public static MySingleton getInstance() { 
		return INSTANCE;
	}
	
    private Object readResolve() throws ObjectStreamException {     
      return INSTANCE;
   }
}  

 

 当把 MySingleton对象(通过getInstance方法获得的那个单例对象)序列化后再从内存中读出时, 就有一个全新但跟原来一样的MySingleton对象存在了. 那怎么来维护单例模式呢?这就要用到readResolve方法了.

 这样当JVM从内存中反序列化地"组装"一个新对象时,就会自动调用这个 readResolve方法来返回我们指定好的对象了, 单例规则也就得到了保证.

 

 

java中readResolve方法的使用

我们知道java 对象的序列化操作是实现Serializable接口,我们就可以把它往内存地写再从内存里读出而"组装"成一个跟原来一模一样的对象. 但是当我们遇到单例序列化的时候,就出现问题了。当从内存读出而组装的对象破坏了单例的规则,会创建新的对象。单例要求JVM只有一个对象,而通过反序化时就会产生一个克隆的对象,这就打破了单例的规则。

 

 

 Java:单例模式的七种写法 
http://icewubin.iteye.com/blog/256869

分享到:
评论

相关推荐

    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