当前位置:首页 > 开发 > 编程语言 > 编程 > 正文

实现严谨的singleton类

发表于: 2012-03-08   作者:coolxing   来源:转载   浏览次数:
摘要: [coolxing按: 转载请注明作者和出处, 如有谬误, 欢迎在评论中指正.]  singleton模式是大多数javaer耳熟能详的, 不过要做到真正的单例其实很不容易, 你需要考虑以下问题: 1. 延迟加载时多线程环境下是否能保证单例? 2. 是否可以通过暴力反射获得新的对象? 3. 是否可以通过clone方法获得新的对象? 4. 是否可以通过序列化获得新的对象? &

[coolxing按: 转载请注明作者和出处, 如有谬误, 欢迎在评论中指正.] 

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

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

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

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

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

 

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

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

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

 

public enum Weekday {
	MONDAY;
	private Weekday() {}
}

 

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

 

暴力反射

测试代码如下:

public static void main(String[] args) throws Exception {
		Class<?> clazz = Weekday.class;
		Constructor<?> cc = clazz.getDeclaredConstructor(null);
		cc.setAccessible(true);
		cc.newInstance(null);
}

 

运行结果是抛出异常: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方法如下:

protected final Object clone() throws CloneNotSupportedException {
	throw new CloneNotSupportedException();
}

 

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

 

序列化

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

private void readObject(ObjectInputStream in) throws IOException,
        ClassNotFoundException {
            throw new InvalidObjectException("can't deserialize enum");
}

private void readObjectNoData() throws ObjectStreamException {
        throw new InvalidObjectException("can't deserialize enum");
}
 

同暴力反射一样, 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的最简单, 最严谨的方式.

Java Language Specification下载路径:http://docs.oracle.com/javase/specs/

 

实现严谨的singleton类

  • 0

    开心

    开心

  • 0

    板砖

    板砖

  • 0

    感动

    感动

  • 0

    有用

    有用

  • 0

    疑问

    疑问

  • 0

    难过

    难过

  • 0

    无聊

    无聊

  • 0

    震惊

    震惊

编辑推荐
Singleton模式C++实现 Singleton是设计模式中比较简单的一个。园中的朋友们应该都很熟悉了。前段时
Scala比Java更为面向对象的特点之一是scala不能定义静态成员,而是代之以定义单例对象。除了用objec
Singleton是设计模式中比较简单的一个。园中的朋友们应该都很熟悉了。前段时间参加xxx外企的面试,
http://www.ibm.com/developerworks/cn/java/l-jdkdp/part2/ 简介: 在 上一部分的内容中,我们讲到
在软件设计过程中,根据需求需要实现singleton单例模型。比如创建数据库的连接实例。下面利用python
题目:设计一个类,我们只能生成该类的一个实例 public class Test02 { /** * 单例模式,懒汉式,线
前阵子写静态lib导出单实例多线程安全API时,出现了CRITICAL_SECTION初始化太晚的问题,之后查看了
个人认为 Singleton 模式是设计模式中最为简单、最为常见、最容易实现,也是最应该 熟悉和掌握的模
再谈Singleton 前些时候写了一篇关于Singleton模式的使用心得,发布在这个页面: http://blog.csdn.
1 #include <iostream> 2 #include <vector> 3 using namespace std; 4 5 template <
版权所有 IT知识库 CopyRight © 2009-2015 IT知识库 IT610.com , All Rights Reserved. 京ICP备09083238号