Java反射破坏单例模式

今天电话面试的时候问到了,Google了一下

原出处:

http://blog.csdn.net/lws332969674/article/details/8125893

一、 Java中的反射技术可以获取类的所有方法、成员变量、还能访问private的构造方法,这样一来,单例模式中用的私有构造函数被调用就会产生多个实例,编写代码测试一下。

[java] view plaincopyprint?

  1. package test;
  2. import java.lang.reflect.Constructor;
  3. public class SingetonTest {
  4. private static SingetonTest singleton = null;
  5. private int s = 0;
  6. // 构造方法是私有的
  7. private SingetonTest(){}
  8. // 同步的获取实例方法
  9. public static synchronized SingetonTest getInstance(){
  10. // 懒汉模式的单例方法
  11. if(null == singleton){
  12. singleton = new SingetonTest();
  13. }
  14. return singleton;
  15. }
  16. public int getS() {
  17. return s;
  18. }
  19. public void setS(int s) {
  20. this.s = s;
  21. }
  22. /**
  23. * @param args
  24. */
  25. public static void main(String[] args) {
  26. try {
  27. Constructor con = SingetonTest.class.getDeclaredConstructor();
  28. con.setAccessible(true);
  29. // 通过反射获取实例
  30. SingetonTest singetonTest1 = (SingetonTest)con.newInstance();
  31. SingetonTest singetonTest2 = (SingetonTest)con.newInstance();
  32. // 常规方法获取实例
  33. SingetonTest singetonTest3 = SingetonTest.getInstance();
  34. SingetonTest singetonTest4 = SingetonTest.getInstance();
  35. // 测试输出
  36. System.out.println("singetonTest1.equals(singetonTest2) :" +  singetonTest1.equals(singetonTest2));
  37. System.out.println("singetonTest3.equals(singetonTest4) :" +  singetonTest3.equals(singetonTest4));
  38. System.out.println("singetonTest1.equals(singetonTest3) :" +  singetonTest1.equals(singetonTest3));
  39. singetonTest1.setS(1);
  40. singetonTest2.setS(2);
  41. singetonTest3.setS(3);
  42. singetonTest4.setS(4);
  43. System.out.println("1:" + singetonTest1.getS() + "  2:" + singetonTest2.getS()+ "  3:" + singetonTest3.getS()+ "  4:" + singetonTest4.getS());
  44. } catch (Exception e) {
  45. // TODO Auto-generated catch block
  46. e.printStackTrace();
  47. }
  48. }
  49. }

测试结果:

singetonTest1.equals(singetonTest2) :false
singetonTest3.equals(singetonTest4) :true
singetonTest1.equals(singetonTest3) :false
1:1  2:2  3:4  4:4

通过反射技术生成的两个实例不同,通过常规方法获取的两个实例相同(即同一个实例,单例)。

二、防止反射破坏单例模式,构造函数调用时进行处理,当构造函数第2次被调用时抛出异常!修改构造方法如下:

[java] view plaincopyprint?

  1. private static boolean flag = false;
  2. // 构造方法是私有的
  3. private SingetonTest(){
  4. if(flag){
  5. flag = !flag;
  6. }
  7. else{
  8. try {
  9. throw new Exception("duplicate instance create error!" + SingetonTest.class.getName());
  10. } catch (Exception e) {
  11. // TODO Auto-generated catch block
  12. e.printStackTrace();
  13. }
  14. }
  15. }

三、一些思考

1)单例模式是为了保证一个类只有一个实例,整个系统只能有自己创建的一个实例。应用在数据库连接或单个队列处理等问题。

2)单例模式构造,懒汉模式以时间换空间(用的时候才生产实例,每次都判断);懒汉模式以空间换时间,直接创建实例,以后不用判断直接用。

3)懒汉模式线程安全问题,获取实例的方法要加上synchronized进行同步。

4)java的反射技术不要用于实例创建,反射主要用于Spring的IOC, Hibernate和白盒测试。

---------------------------------------------------------我是枚举分割线-------------------------------------------------------

可以用枚举防止反射破坏单例模型(PS:JAVA 枚举是线程安全的)

From:http://blog.csdn.net/tounaobun/article/details/8514989

JDK1.5及以后,增加了实现Singleton的第三种方法。只需编写一个包含单个元素的枚举类型。

Java代码

  1. public enum Adam {
  2. INSTANCE;   //只有一个元素
  3. public void leaveTheBuilding() {;;;}
  4. }

使用这种方法的好处是可以防止多次实例化,无偿提供了序列化机制,即使是面对复杂的序列化或者反射公鸡。

当你试图通过反射调用枚举类型的构造器时(默认构造器为private),如果调用了setAccessible(true)方法,将会抛出IllegalArgumentException:

Java代码

  1. Exception in thread "main" java.lang.IllegalArgumentException: Cannot reflectively create enum objects
  2. at java.lang.reflect.Constructor.newInstance(Unknown Source)
  3. at org.reflect.Test.main(Test.java:23)

对于前两种单例模式,为了使Singleton能够序列化,除了实现标记接口Serializable外,还需增加类似下面的方法,防止反序列化时生成“假冒”的单例类:

Java代码

  1. private Object readResolve() {
  2. return INSTANCE;
  3. }

而对于枚举类型,完全不用多此一举。因为枚举类型已经提供了该机制。

时间: 2024-08-18 04:44:14

Java反射破坏单例模式的相关文章

单例模式--反射--防止序列化破坏单例模式

本文牵涉到的概念: 1.单例模式------唯一最佳实现方式,使用枚举类实现 2.单例模式的几种实现,各自的缺点 3.反射;反射是如何破坏单例模式 4.序列化:序列化如何破坏单例模式 单例模式 单例模式,是指在任何时候,该类只能被实例化一次,在任何时候,访问该类的对象,对象都是同一的,只有一个. 单例模式的实现方式: a.使用类公有的静态成员来保存该唯一对象 public class EagerSingleton { // jvm保证在任何线程访问uniqueInstance静态变量之前一定先创

Java反射

1. 介绍 反射是一种能够在程序运行时动态访问.修改某个类中任意属性和方法的机制. 具体:对于任意一个类,都能够知道这个类的所有属性和方法对于任意一个对象,都能够调用它的任意一个方法和属性 在运行时,当加载完类之后,JVM在堆内存中会自动产生一个Class类型的对象,这个对象包含了完整的类的结构信息 这个Class对象就像一面镜子,透过这个镜子看到类的结构 那么,如何得到这个Class对象呢?以下可否 Class c = new Class(); 答案是不行的,因为Class的构造函数定义为私有

[转]JAVA设计模式之单例模式

原文地址:http://blog.csdn.net/jason0539/article/details/23297037 概念: java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍三种:懒汉式单例.饿汉式单例.登记式单例. 单例模式有以下特点: 1.单例类只能有一个实例. 2.单例类必须自己创建自己的唯一实例. 3.单例类必须给所有其他对象提供这一实例. 单例模式确保某个类只有一个实例,而且自行实例化并向整个系统提供这个实例.在计算机系统中,线程池.缓存.日志对象.对话

Java 设计模式(3)单例模式

前言 概念: java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍三种:懒汉式单例.饿汉式单例.登记式单例. 单例模式有以下特点: 1.单例类只能有一个实例. 2.单例类必须自己创建自己的唯一实例. 3.单例类必须给所有其他对象提供这一实例. 单例模式确保某个类只有一个实例,而且自行实例化并向整个系统提供这个实例.在计算机系统中,线程池.缓存.日志对象.对话框.打印机.显卡的驱动程序对象常被设计成单例.这些应用都或多或少具有资源管理器的功能.每台计算机可以有若干个打印机,

JAVA设计模式之单例模式(转)

本文继续介绍23种设计模式系列之单例模式. 概念: java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍三种:懒汉式单例.饿汉式单例.登记式单例. 单例模式有以下特点: 1.单例类只能有一个实例. 2.单例类必须自己创建自己的唯一实例. 3.单例类必须给所有其他对象提供这一实例. 单例模式确保某个类只有一个实例,而且自行实例化并向整个系统提供这个实例.在计算机系统中,线程池.缓存.日志对象.对话框.打印机.显卡的驱动程序对象常被设计成单例.这些应用都或多或少具有资源管理器

Java反射及其在Android中的应用学习总结

一. Java反射机制 Reflection 是Java被视为动态(或准动态)语言的一个关键性质.这个机制同意程序在执行时透过Reflection APIs取得不论什么一个已知名称的class的内部信息,包含其modifiers(诸如public, static 等等).superclass(比如Object).实现之interfaces(比如Serializable).也包含fields和methods的全部信息,并可于执行时改变fields内容或调用methods(包含被声明为private

通过Java反射测试类私有成员

在Java开发阶段,因为追求架构规范和遵循设计原则,所以要用private和protected修饰符去定义类的成员方法.变量.常量,这使得代码具封装性.内聚性等,但在测试阶段会造成一定的不便.通过Java的反射机制,便能很好地解决该问题. ReflectUtil.java //...... /** * @author yumin * @since 2015-03-02 14:52 */ public class ReflectUtil { private ReflectUtil() { } //

Java反射初探 ——“当类也学会照镜子”

反射的作用 开门见山地说说反射的作用 1.为我们提供了全面的分析类信息的能力 2.动态加载类 我理解的“反射”的意义 (仅个人理解哈) 我理解的java反射机制就是: 提供一套完善而强大的API“反射“类的结构. 打个比方,反射机制就像是一面镜子,而类就像是一个在照着镜子的人. 镜子(反射机制)照出(反射)了人的全貌(类的全方位的信息,例如方法,成员变量和构造器等的相关信息) 为什么要照镜子? 因为不照镜子看不清楚自己的全貌,“镜子”就是为了解决这个问题出现的(为我们提供全面分析类的能力) 好吧

小怪兽 Java反射技术 等你来打

Java反射技术,是java的难点,也是程序员进化过程中的必打小怪兽,这里就根据java api好好研究一下java的反射技术. Class Fields Methods 请先回忆一下[java类的定义]. 一.反射是什么 反射是什么?百度百科这样定义:JAVA反射机制是在运行状态下,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取对象信息以及动态调用对象方法的功能称为java语言的反射机制. 可以这样理解,大家小时候有木有玩过平