A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© dftgg 中级黑马   /  2014-9-2 15:34  /  3770 人查看  /  6 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

1)枚举单例模式代码简洁

       这是迄今为止最大的优点,如果你曾经在java5之前写过单例模式实现代码,那么你会知道即使是使用双检锁你有时候也会返回不止一个实例对象。虽然这种问题通过改善java内存模型和使用volatile变量可以解决,但是这种方法对于很多初学者来说写起来还是很棘手。相比用synchronization的双检索实现方式来说,枚举单例就简单多了。你不相信?比较一下下面的双检索实现代码和枚举实现代码就知道了。

用枚举实现的单例:

这是我们通常写枚举单例的方式,它可能包含实例变量和实例方法,但是简单来说我什么都没用,需要注意的是如果你使用实例方法,你就需要确保方法的线程安全性,避免它会影响对象的状态。通常情况下枚举里面创建实例是线程安全的,但是其它的方法就需要编程者自己去考虑了。

/**

* Singleton   pattern   example using Java Enumj

*/

public   enum   EasySingleton {
    INSTANCE ;
}

代码就这么简单,你可以使用EasySingleton.INSTANCE调用它,比起你在单例中调用getInstance()方法容易多了。

用双检索实现单例:

下面的代码是用双检索实现单例模式的例子,在这里getInstance()方法检查了两次来判断INSTANCE是否为null,这就是为什么叫双检索的原因,记住双检索在java5之前是有问题的,但是java5在内存模型中有了volatile变量之后就没问题了。

/**

* Singleton   pattern   example with Double checked Locking

*/

public   class   DoubleCheckedLockingSingleton {       private   volatile   DoubleCheckedLockingSingleton INSTANCE ;     
      private   DoubleCheckedLockingSingleton (){}     
      public   DoubleCheckedLockingSingleton getInstance (){           if ( INSTANCE   ==   null ){               synchronized ( DoubleCheckedLockingSingleton. class ){                   //double checking Singleton instance                   if ( INSTANCE   ==   null ){                     INSTANCE   =   new   DoubleCheckedLockingSingleton () ;                   }               }           }           return   INSTANCE ;       } }
你可以访问DoubleCheckedLockingSingleTon.getInstance()来获得实例对象。

现在看看两者创建一个懒加载线程安全的单例需要的代码数量。使用枚举单例模式你只需要一行代码搞定因为枚举实例的创建是线程安全的。

你可能会说比起使用双检索方法还有更好的方法实现单例模式,但是任何一种方法都有它的利和弊,就像我下面例子中展示的我很喜欢的一种在类加载期间初始化静态域的单例实现方式,但是要记住这不是一种懒加载单例方式。

用静态工厂方法实现单例:

这是java中我比较喜欢的一种实现单例模式的方法,由于单例实例是static和final的,当类第一次被加载到内存它就实例化了,所以这种实例的创建方式是线程安全的。

/**
* Singleton   pattern   example with static factory method

*/ public   class   Singleton {
      //initailzed during class loading
      private   static   final   Singleton INSTANCE   =   new   Singleton () ;
   
      //to prevent creating another instance of Singleton
      private   Singleton (){}
      public   static   Singleton getSingleton (){
          return   INSTANCE ;
      }
}
你可以调用Singleton.getInstance()方法来获得实例对象。

2)枚举单例可以自己处理序列化

传统的单例模式的另外一个问题是一旦你实现了serializable接口,他们就不再是单例的了,因为readObject()方法总是返回一个新的实例对象,就像java中的构造器一样。你可以使用readResolve()方法来避免这种情况,通过像下面的例子中这样用单例来替换新创建的实例:

//readResolve to prevent another instance of Singleton       private   Object readResolve (){           return   INSTANCE ;       }

如果你的单例类包含状态的话就变的更复杂了,你需要把他们置为transient状态,但是用枚举单例的话,序列化就不要考虑了。

3)枚举单例是线程安全的

就像第一点提到的,由于枚举实例的创建默认就是线程安全的,你不需要担心双检锁。

6 个回复

倒序浏览
非常好,学习了
回复 使用道具 举报
努力学习中  感谢分享
回复 使用道具 举报
看看,学习下。
回复 使用道具 举报
恩,重点啊
回复 使用道具 举报
给力!!!!
回复 使用道具 举报
顶一个.......
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马