实现严谨的singleton类

singleton模式是大多数javaer耳熟能详的, 不过要做到真正的单例其实很不容易, 你需要考虑以下问题:

1. 延迟加载时多线程环境下是否能保证单例?

2. 是否可以通过暴力反射获得新的对象?

3. 是否可以通过clone方法获得新的对象?

4. 是否可以通过序列化获得新的对象?

 

对于问题1, 可以通过双重检查加锁解决. 这是运用单例的常识, 不再详细说明.

问题2, 3, 4则是很难避免的.

但是我们可以通过java提供的一种简单的方式创建一个真正的singleton类: 仅有一个实例的枚举类型. 如:

 

Java代码   收藏代码
  1. public enum Weekday {  
  2.     MONDAY;  
  3.     private Weekday() {}  
  4. }  

 

下面我们对这个枚举类进行一一验证.

 

暴力反射

测试代码如下:

Java代码   收藏代码
  1. public static void main(String[] args) throws Exception {  
  2.         Class<?> clazz = Weekday.class;  
  3.         Constructor<?> cc = clazz.getDeclaredConstructor(null);  
  4.         cc.setAccessible(true);  
  5.         cc.newInstance(null);  
  6. }  

 

运行结果是抛出异常:Exception in thread "main" java.lang.NoSuchMethodException: cn.xing.test.Weekday.<init>()

明明Weekday有一个无参的构造函数, 为何不能通过暴力反射访问?

最新的Java Language Specification (§8.9)规定:  Reflective instantiation of enum types is prohibited. 这是java语言的内置规范.

 

 

clone方法

所有的枚举类都继承自java.lang.Enum类, 而不是Object类. 在java.lang.Enum类中clone方法如下:

Java代码   收藏代码
  1. protected final Object clone() throws CloneNotSupportedException {  
  2.     throw new CloneNotSupportedException();  
  3. }  

 

调用该方法将抛出异常, 且final意味着子类不能重写clone方法, 所以通过clone方法获取新的对象是不可取的.

 

序列化

java.lang.Enum类的readObject方法如下:

Java代码   收藏代码
  1. private void readObject(ObjectInputStream in) throws IOException,  
  2.         ClassNotFoundException {  
  3.             throw new InvalidObjectException("can't deserialize enum");  
  4. }  
  5.   
  6. private void readObjectNoData() throws ObjectStreamException {  
  7.         throw new InvalidObjectException("can't deserialize enum");  
  8. }  
 

同暴力反射一样, Java Language Specification (§8.9)有着这样的规定: the special treatment by the serialization mechanism ensures that duplicate instances are never created as a result of deserialization.

 

综上所述, 创建仅有一个实例的枚举类型是实现singleton的最简单, 最严谨的方式.

原文地址:http://coolxing.iteye.com/blog/1446648

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值