JAVA反射机制

原文链接:http://blog.csdn.net/justinavril/article/details/2873664

Reflection是Java 程序开发语言的特征之一,它允许运行中的 Java 程序对自身进行检查,或者说"自审",并能直接操作程序的内部属性。例如,使用它能获得 Java 类中各成员的名称并显示出来。 Java 的这一能力在实际应用中也许用得不是很多,但是在其它的程序设计语言中根本就不存在这一特性。例如,Pascal、C 或者 C++ 中就没有办法在程序中获得函数定义相关的信息。

JavaBean 是 reflection 的实际应用之一,它能让一些工具可视化的操作软件组件。这些工具通过 reflection 动态的载入并取得 Java 组件(类) 的属性。

1. 一个简单的例子

考虑下面这个简单的例子,让我们看看 reflection 是如何工作的。

  1. import java.lang.reflect.*; 
  2. public class DumpMethods { 
  3.    public static void main(String args[]) { 
  4.       try { 
  5.            Class c = Class.forName("java.util.Stack"); 
  6.            Method m[] = c.getDeclaredMethods(); 
  7.             
  8.            for (int i = 0; i < m.length; i++) 
  9.                System.out.println(m[i].toString()); 
  10.       } 
  11.       catch (Throwable e){ 
  12.             System.err.println(e); 
  13.       } 
  14.    } 
  15. }

它的结果输出为:

  1. public synchronized java.lang.Object java.util.Stack.pop()
  2. public java.lang.Object java.util.Stack.push(java.lang.Object)
  3. public boolean java.util.Stack.empty()
  4. public synchronized java.lang.Object java.util.Stack.peek()
  5. public synchronized int java.util.Stack.search(java.lang.Object)

这样就列出了java.util.Stack 类的各方法名以及它们的限制符和返回类型。

这个程序使用 Class.forName 载入指定的类,然后调用 getDeclaredMethods 来获取这个类中定义了的方法列表。java.lang.reflect.Methods 是用来描述某个类中单个方法的一个类。

2.开始使用 Reflection

用于 reflection 的类,如 Method,可以在 java.lang.relfect 包中找到。使用这些类的时候必须要遵循三个步骤:第一步是获得你想操作的类的 java.lang.Class 对象。在运行中的 Java 程序中,用 java.lang.Class 类来描述类和接口等。

下面就是获得一个 Class 对象的方法之一:

Class c = Class.forName("java.lang.String");

这条语句得到一个 String 类的类对象。还有另一种方法,如下面的语句:

Class c = int.class; 或者 Class c = Integer.TYPE;

它们可获得基本类型的类信息。其中后一种方法中访问的是基本类型的封装类 (如 Integer) 中预先定义好的 TYPE 字段。

第二步是调用诸如 getDeclaredMethods 的方法,以取得该类中定义的所有方法的列表。

一旦取得这个信息,就可以进行第三步了——使用 reflection API 来操作这些信息,如下面这段代码:

  1. Class c = Class.forName("java.lang.String"); 
  2. Method m[] = c.getDeclaredMethods(); 
  3. System.out.println(m[0].toString()); 

它将以文本方式打印出 String 中定义的第一个方法的原型。

在下面的例子中,这三个步骤将为使用 reflection 处理特殊应用程序提供例证。

模拟 instanceof 操作符

得到类信息之后,通常下一个步骤就是解决关于 Class 对象的一些基本的问题。例如,Class.isInstance 方法可以用于模拟 instanceof 操作符:

  1. class S { 
  2. }  
  3. public class IsInstance { 
  4.    public static void main(String args[]) { 
  5.       try { 
  6.            Class cls = Class.forName("S"); 
  7.            boolean b1 = cls.isInstance(new Integer(37)); 
  8.            System.out.println(b1); 
  9.            boolean b2 = cls.isInstance(new S()); 
  10.            System.out.println(b2); 
  11.       } 
  12.       catch (Throwable e) { 
  13.            System.err.println(e); 
  14.       } 
  15.    } 
  16. }

在这个例子中创建了一个S 类的 Class 对象,然后检查一些对象是否是S的实例。Integer(37) 不是,但 new S()是。

3.找出类的方法

找出一个类中定义了些什么方法,这是一个非常有价值也非常基础的 reflection 用法。下面的代码就实现了这一用法:

  1. import java.lang.reflect.*;  
  2. public class Method1 { 
  3.    private int f1(Object p, int x) throws NullPointerException { 
  4.    if (p == null
  5.       throw new NullPointerException(); 
  6.    return x; 
  7. }  
  8.    public static void main(String args[]) { 
  9.        try { 
  10.            Class cls = Class.forName("Method1"); 
  11.            Method methlist[] = cls.getDeclaredMethods(); 
  12.            for (int i = 0; i < methlist.length; i++) { 
  13.                Method m = methlist[i]; 
  14.                System.out.println("name = " + m.getName()); 
  15.                System.out.println("decl class = " + m.getDeclaringClass()); 
  16.                Class pvec[] = m.getParameterTypes(); 
  17.                for (int j = 0; j < pvec.length; j++) 
  18.                    System.out.println("param #" + j + " " + pvec[j]); 
  19.                Class evec[] = m.getExceptionTypes(); 
  20.                for (int j = 0; j < evec.length; j++) 
  21.                    System.out.println("exc #" + j + " " + evec[j]); 
  22.                System.out.println("return type = " + m.getReturnType()); 
  23.                System.out.println("-----"); 
  24.            } 
  25.        } 
  26.        catch (Throwable e) { 
  27.            System.err.println(e); 
  28.        } 
  29.    } 
  30. }  

这个程序首先取得 method1 类的描述,然后调用 getDeclaredMethods 来获取一系列的 Method 对象,它们分别描述了定义在类中的每一个方法,包括 public 方法、protected 方法、package 方法和 private 方法等。如果你在程序中使用 getMethods 来代替 getDeclaredMethods,你还能获得继承来的各个方法的信息。

取得了 Method 对象列表之后,要显示这些方法的参数类型、异常类型和返回值类型等就不难了。这些类型是基本类型还是类类型,都可以由描述类的对象按顺序给出。

输出的结果如下:

  1. name = f1 
  2. decl class = class method1 
  3. param #0 class java.lang.Object 
  4. param #1 int 
  5. exc #0 class java.lang.NullPointerException 
  6. return type = int
  7. -----
  8. name = main 
  9. decl class = class method1 
  10. param #0 class [Ljava.lang.String; 
  11. return type = void 

4.获取构造器信息

获取类构造器的用法与上述获取方法的用法类似,如:

  1. import java.lang.reflect.*;
  2. public class Constructor1 { 
  3.    public Constructor1() { 
  4.    }  
  5.    protected Constructor1(int i, double d) { 
  6.    }  
  7.    public static void main(String args[]) { 
  8.       try { 
  9.            Class cls = Class.forName("Constructor1"); 
  10.            Constructor ctorlist[] = cls.getDeclaredConstructors(); 
  11.            for (int i = 0; i < ctorlist.length; i++) { 
  12.               Constructor ct = ctorlist[i]; 
  13.               System.out.println("name = " + ct.getName()); 
  14.               System.out.println("decl class = " + ct.getDeclaringClass()); 
  15.               Class pvec[] = ct.getParameterTypes(); 
  16.               for (int j = 0; j < pvec.length; j++) 
  17.                  System.out.println("param #" + j + " " + pvec[j]); 
  18.               Class evec[] = ct.getExceptionTypes(); 
  19.               for (int j = 0; j < evec.length; j++) 
  20.                  System.out.println("exc #" + j + " " + evec[j]); 
  21.               System.out.println("-----"); 
  22.            } 
  23.       } 
  24.       catch (Throwable e) { 
  25.            System.err.println(e); 
  26.       } 
  27.    } 

这个例子中没能获得返回类型的相关信息,那是因为构造器没有返回类型。

这个程序运行的结果是:

  1. name = Constructor1
  2. decl class = class Constructor1
  3. param #0 int
  4. param #1 double
  5. -----
  6. name = Constructor1
  7. decl class = class Constructor1
  8. -----

5.获取类的字段(域)

找出一个类中定义了哪些数据字段也是可能的,下面的代码就在干这个事情:

  1. import java.lang.reflect.*;  
  2. public class Field1 { 
  3.    private double d; 
  4.    public static final int i = 37; 
  5.    String s = "testing";  
  6.    public static void main(String args[]) { 
  7.       try { 
  8.            Class cls = Class.forName("Field1"); 
  9.            Field fieldlist[] = cls.getDeclaredFields(); 
  10.            for (int i = 0; i < fieldlist.length; i++) { 
  11.               Field fld = fieldlist[i]; 
  12.               System.out.println("name = " + fld.getName()); 
  13.               System.out.println("decl class = " + fld.getDeclaringClass()); 
  14.               System.out.println("type = " + fld.getType()); 
  15.               int mod = fld.getModifiers(); 
  16.               System.out.println("modifiers = " + Modifier.toString(mod)); 
  17.               System.out.println("-----"); 
  18.            } 
  19.       } 
  20.       catch (Throwable e) { 
  21.            System.err.println(e); 
  22.       } 
  23.    } 
  24. }  

这个例子和前面那个例子非常相似。例中使用了一个新东西 Modifier,它也是一个 reflection 类,用来描述字段成员的修饰语,如“private int”。这些修饰语自身由整数描述,而且使用 Modifier.toString 来返回以“官方”顺序排列的字符串描述 (如“static”在“final”之前)。这个程序的输出是:

  1. name = d
  2. decl class = class Field1
  3. type = double
  4. modifiers = private
  5. -----
  6. name = i
  7. decl class = class Field1
  8. type = int
  9. modifiers = public static final
  10. -----
  11. name = s
  12. decl class = class Field1
  13. type = class java.lang.String
  14. modifiers = 
  15. -----

和获取方法的情况一下,获取字段的时候也可以只取得在当前类中申明了的字段信息 (getDeclaredFields),或者也可以取得父类中定义的字段 (getFields) 。

6.根据方法的名称来执行方法

文本到这里,所举的例子无一例外都与如何获取类的信息有关。我们也可以用 reflection 来做一些其它的事情,比如执行一个指定了名称的方法。下面的示例演示了这一操作:

  1. import java.lang.reflect.*; 
  2. public class Method2 { 
  3.    public int add(int a, int b) { 
  4.       return a + b; 
  5.    } 
  6.    public static void main(String args[]) { 
  7.       try { 
  8.            Class cls = Class.forName("Method2"); 
  9.            Class partypes[] = new Class[2]; 
  10.            partypes[0] = Integer.TYPE; 
  11.            partypes[1] = Integer.TYPE; 
  12.        
  13.            Method meth = cls.getMethod("add", partypes); 
  14.            Method2 methobj = new Method2(); 
  15.            Object arglist[] = new Object[2]; 
  16.            arglist[0] = new Integer(37); 
  17.            arglist[1] = new Integer(47); 
  18.            Object retobj = meth.invoke(methobj, arglist); 
  19.            Integer retval = (Integer) retobj; 
  20.            System.out.println(retval.intValue()); 
  21.       } 
  22.       catch (Throwable e) { 
  23.            System.err.println(e); 
  24.       } 
  25.    } 
  26. }

假如一个程序在执行的某处的时候才知道需要执行某个方法,这个方法的名称是在程序的运行过程中指定的 (例如,JavaBean 开发环境中就会做这样的事),那么上面的程序演示了如何做到。

上例中,getMethod用于查找一个具有两个整型参数且名为 add 的方法。找到该方法并创建了相应的Method 对象之后,在正确的对象实例中执行它。执行该方法的时候,需要提供一个参数列表,这在上例中是分别包装了整数 37 和 47 的两个 Integer 对象。执行方法的返回的同样是一个 Integer 对象,它封装了返回值 84。

7.创建新的对象

对于构造器,则不能像执行方法那样进行,因为执行一个构造器就意味着创建了一个新的对象 (准确的说,创建一个对象的过程包括分配内存和构造对象)。所以,与上例最相似的例子如下:

  1. import java.lang.reflect.*;  
  2. public class Constructor2 { 
  3.    public Constructor2() { 
  4.    }  
  5.    public Constructor2(int a, int b) { 
  6.       System.out.println("a = " + a + " b = " + b); 
  7.    }  
  8.    public static void main(String args[]) { 
  9.       try { 
  10.            Class cls = Class.forName("Constructor2"); 
  11.            Class partypes[] = new Class[2]; 
  12.            partypes[0] = Integer.TYPE; 
  13.            partypes[1] = Integer.TYPE; 
  14.            Constructor ct = cls.getConstructor(partypes); 
  15.            Object arglist[] = new Object[2]; 
  16.            arglist[0] = new Integer(37); 
  17.            arglist[1] = new Integer(47); 
  18.            Object retobj = ct.newInstance(arglist); 
  19.       } 
  20.       catch (Throwable e) { 
  21.            System.err.println(e); 
  22.       } 
  23.    } 

根据指定的参数类型找到相应的构造函数并执行它,以创建一个新的对象实例。使用这种方法可以在程序运行时动态地创建对象,而不是在编译的时候创建对象,这一点非常有价值。

8.改变字段(域)的值

reflection 的还有一个用处就是改变对象数据字段的值。reflection 可以从正在运行的程序中根据名称找到对象的字段并改变它,下面的例子可以说明这一点:

  1. import java.lang.reflect.*;  
  2. public class Field2 { 
  3.    public double d;  
  4.    public static void main(String args[]) { 
  5.       try { 
  6.            Class cls = Class.forName("Field2"); 
  7.            Field fld = cls.getField("d"); 
  8.            Field2 f2obj = new Field2(); 
  9.            System.out.println("d = " + f2obj.d); 
  10.            fld.setDouble(f2obj, 12.34); 
  11.            System.out.println("d = " + f2obj.d); 
  12.       } 
  13.       catch (Throwable e) { 
  14.            System.err.println(e); 
  15.       } 
  16.    } 

这个例子中,字段 d 的值被变为了 12.34。

9.使用数组

本文介绍的 reflection 的最后一种用法是创建的操作数组。数组在 Java 语言中是一种特殊的类类型,一个数组的引用可以赋给 Object 引用。观察下面的例子看看数组是怎么工作的:

  1. import java.lang.reflect.*;  
  2. public class Array1 { 
  3.    public static void main(String args[]) { 
  4.       try { 
  5.            Class cls = Class.forName("java.lang.String"); 
  6.            Object arr = Array.newInstance(cls, 10); 
  7.            Array.set(arr, 5"this is a test"); 
  8.            String s = (String) Array.get(arr, 5); 
  9.            System.out.println(s); 
  10.       } 
  11.       catch (Throwable e) { 
  12.            System.err.println(e); 
  13.       } 
  14.    } 

例中创建了 10 个单位长度的 String 数组,为第 5 个位置的字符串赋了值,最后将这个字符串从数组中取得并打印了出来。

下面这段代码提供了一个更复杂的例子:

  1. import java.lang.reflect.*;  
  2. public class Array2 { 
  3.    public static void main(String args[]) { 
  4.       int dims[] = new int[]{51015}; 
  5.       Object arr = Array.newInstance(Integer.TYPE, dims); 
  6.       Object arrobj = Array.get(arr, 3); 
  7.       Class cls = arrobj.getClass().getComponentType(); 
  8.       System.out.println(cls); 
  9.       arrobj = Array.get(arrobj, 5); 
  10.       Array.setInt(arrobj, 1037); 
  11.       int arrcast[][][] = (int[][][]) arr; 
  12.       System.out.println(arrcast[3][5][10]); 
  13.    } 

例中创建了一个 5 x 10 x 15 的整型数组,并为处于 [3][5][10] 的元素赋了值为 37。注意,多维数组实际上就是数组的数组,例如,第一个 Array.get 之后,arrobj 是一个 10 x 15 的数组。进而取得其中的一个元素,即长度为 15 的数组,并使用 Array.setInt 为它的第 10 个元素赋值。

注意创建数组时的类型是动态的,在编译时并不知道其类型。

-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

原文链接:http://lavasoft.blog.51cto.com/62575/43218/

Java反射机制深入研究
 
Java 反射是Java语言的一个很重要的特征,它使得Java具体了“动态性”。
 
在Java运行时环境中,对于任意一个类,能否知道这个类有哪些属性和方法?对于任意一个对象,能否调用它的任意一个方法?答案是肯定的。这种动态获取类的信息以及动态调用对象的方法的功能来自于Java 语言的反射(Reflection)机制。
 
Java 反射机制主要提供了以下功能:
在运行时判断任意一个对象所属的类。
在运行时构造任意一个类的对象。
在运行时判断任意一个类所具有的成员变量和方法。
在运行时调用任意一个对象的方法。
 
Reflection 是Java被视为动态(或准动态)语言的一个关键性质。这个机制允许程序在运行时透过Reflection APIs取得任何一个已知名称的class的内部信息,包括其modifiers(诸如public, static 等等)、superclass(例如Object)、实现之interfaces(例如Serializable),也包括fields和methods的所有信息,并可于运行时改变fields内容或调用methods。
 
一般而言,开发者社群说到动态语言,大致认同的一个定义是:“程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言”。从这个观点看,Perl,Python,Ruby是动态语言,C++,Java,C#不是动态语言。
 
尽管在这样的定义与分类下Java不是动态语言,它却有着一个非常突出的动态相关机制:Reflection。这个字的意思是“反射、映象、倒影”,用在Java身上指的是我们可以于运行时加载、探知、使用编译期间完全未知的classes。换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods。这种“看透class”的能力(the ability of the program to examine itself)被称为introspection(内省、内观、反省)。Reflection和introspection是常被并提的两个术语。
 
在JDK中,主要由以下类来实现Java反射机制,这些类都位于java.lang.reflect包中:
Class类:代表一个类。
Field 类:代表类的成员变量(成员变量也称为类的属性)。
Method类:代表类的方法。
Constructor 类:代表类的构造方法。
Array类:提供了动态创建数组,以及访问数组的元素的静态方法。
 
下面给出几个例子看看Reflection API的实际运用:
 
一、通过Class类获取成员变量、成员方法、接口、超类、构造方法等
 
在java.lang.Object 类中定义了getClass()方法,因此对于任意一个Java对象,都可以通过此方法获得对象的类型。Class类是Reflection API 中的核心类,它有以下方法
getName():获得类的完整名字。
getFields():获得类的public类型的属性。
getDeclaredFields():获得类的所有属性。
getMethods():获得类的public类型的方法。
getDeclaredMethods():获得类的所有方法。
getMethod(String name, Class[] parameterTypes):获得类的特定方法,name参数指定方法的名字,parameterTypes 参数指定方法的参数类型。
getConstructors():获得类的public类型的构造方法。
getConstructor(Class[] parameterTypes):获得类的特定构造方法,parameterTypes 参数指定构造方法的参数类型。
newInstance():通过类的不带参数的构造方法创建这个类的一个对象。
 
下面给出一个综合运用的例子:
 
public class RefConstructor {
 
    public static void main(String args[]) throws Exception {
        RefConstructor ref = new RefConstructor();
        ref.getConstructor();
 
    }
 
    public void getConstructor() throws Exception {
        Class c = null;
        c = Class.forName("java.lang.Long");
        Class cs[] = {java.lang.String.class};
 
        System.out.println("\n-------------------------------\n");
 
        Constructor cst1 = c.getConstructor(cs);
        System.out.println("1、通过参数获取指定Class对象的构造方法:");
        System.out.println(cst1.toString());
 
        Constructor cst2 = c.getDeclaredConstructor(cs);
        System.out.println("2、通过参数获取指定Class对象所表示的类或接口的构造方法:");
        System.out.println(cst2.toString());
 
        Constructor cst3 = c.getEnclosingConstructor();
        System.out.println("3、获取本地或匿名类Constructor 对象,它表示基础类的立即封闭构造方法。");
        if (cst3 != null) System.out.println(cst3.toString());
        else System.out.println("-- 没有获取到任何构造方法!");
 
        Constructor[] csts = c.getConstructors();
        System.out.println("4、获取指定Class对象的所有构造方法:");
        for (int i = 0; i < csts.length; i++) {
            System.out.println(csts[i].toString());
        }
 
        System.out.println("\n-------------------------------\n");
 
        Type types1[] = c.getGenericInterfaces();
        System.out.println("1、返回直接实现的接口:");
        for (int i = 0; i < types1.length; i++) {
            System.out.println(types1[i].toString());
        }
 
        Type type1 = c.getGenericSuperclass();
        System.out.println("2、返回直接超类:");
        System.out.println(type1.toString());
 
        Class[] cis = c.getClasses();
        System.out.println("3、返回超类和所有实现的接口:");
        for (int i = 0; i < cis.length; i++) {
            System.out.println(cis[i].toString());
        }
 
        Class cs1[] = c.getInterfaces();
        System.out.println("4、实现的接口");
        for (int i = 0; i < cs1.length; i++) {
            System.out.println(cs1[i].toString());
        }
 
        System.out.println("\n-------------------------------\n");
 
        Field fs1[] = c.getFields();
        System.out.println("1、类或接口的所有可访问公共字段:");
        for (int i = 0; i < fs1.length; i++) {
            System.out.println(fs1[i].toString());
        }
 
        Field f1 = c.getField("MIN_VALUE");
        System.out.println("2、类或接口的指定已声明指定公共成员字段:");
        System.out.println(f1.toString());
 
        Field fs2[] = c.getDeclaredFields();
        System.out.println("3、类或接口所声明的所有字段:");
        for (int i = 0; i < fs2.length; i++) {
            System.out.println(fs2[i].toString());
        }
 
        Field f2 = c.getDeclaredField("serialVersionUID");
        System.out.println("4、类或接口的指定已声明指定字段:");
        System.out.println(f2.toString());
 
        System.out.println("\n-------------------------------\n");
 
        Method m1[] = c.getMethods();
        System.out.println("1、返回类所有的公共成员方法:");
        for (int i = 0; i < m1.length; i++) {
            System.out.println(m1[i].toString());
        }
 
        Method m2 = c.getMethod("longValue", new Class[]{});
        System.out.println("2、返回指定公共成员方法:");
        System.out.println(m2.toString());
 
    }
}
输出结果:输出结果很长,这里不再给出。
 
 
二、运行时复制对象
 
例程ReflectTester 类进一步演示了Reflection API的基本使用方法。ReflectTester类有一个copy(Object object)方法,这个方法能够创建一个和参数object 同样类型的对象,然后把object对象中的所有属性拷贝到新建的对象中,并将它返回
这个例子只能复制简单的JavaBean,假定JavaBean 的每个属性都有public 类型的getXXX()和setXXX()方法。
 
public class ReflectTester {
    public Object copy(Object object) throws Exception {
        // 获得对象的类型
        Class<?> classType = object.getClass();
        System.out.println("Class:" + classType.getName());
 
        // 通过默认构造方法创建一个新的对象
        Object objectCopy = classType.getConstructor(new Class[]{}).newInstance(new Object[]{});
 
        // 获得对象的所有属性
        Field fields[] = classType.getDeclaredFields();
 
        for (int i = 0; i < fields.length; i++) {
            Field field = fields[i];
 
            String fieldName = field.getName();
            String firstLetter = fieldName.substring(0, 1).toUpperCase();
            // 获得和属性对应的getXXX()方法的名字
            String getMethodName = "get" + firstLetter + fieldName.substring(1);
            // 获得和属性对应的setXXX()方法的名字
            String setMethodName = "set" + firstLetter + fieldName.substring(1);
 
            // 获得和属性对应的getXXX()方法
            Method getMethod = classType.getMethod(getMethodName, new Class[]{});
            // 获得和属性对应的setXXX()方法
            Method setMethod = classType.getMethod(setMethodName, new Class[]{field.getType()});
 
            // 调用原对象的getXXX()方法
            Object value = getMethod.invoke(object, new Object[]{});
            System.out.println(fieldName + ":" + value);
            // 调用拷贝对象的setXXX()方法
            setMethod.invoke(objectCopy, new Object[]{value});
        }
        return objectCopy;
    }
 
    public static void main(String[] args) throws Exception {
        Customer customer = new Customer("Tom", 21);
        customer.setId(new Long(1));
 
        Customer customerCopy = (Customer) new ReflectTester().copy(customer);
        System.out.println("Copy information:" + customerCopy.getId() + " " + customerCopy.getName() + " "
                + customerCopy.getAge());
    }
}
 
class Customer {
    private Long id;
 
    private String name;
 
    private int age;
 
    public Customer() {
    }
 
    public Customer(String name, int age) {
        this.name = name;
        this.age = age;
    }
 
    public Long getId() {
        return id;
    }
 
    public void setId(Long id) {
        this.id = id;
    }
 
    public String getName() {
        return name;
    }
 
    public void setName(String name) {
        this.name = name;
    }
 
    public int getAge() {
        return age;
    }
 
    public void setAge(int age) {
        this.age = age;
    }
}
 
输出结果:
 
Class:com.langsin.reflection.Customer
id:1
name:Tom
age:21
Copy information:1 Tom 21
 
Process finished with exit code 0
 
解说:
ReflectTester 类的copy(Object object)方法依次执行以下步骤
(1)获得对象的类型:
Class classType=object.getClass();
System.out.println("Class:"+classType.getName());
 
(2)通过默认构造方法创建一个新对象:
Object objectCopy=classType.getConstructor(new Class[]{}).newInstance(new Object[]{});
以上代码先调用Class类的getConstructor()方法获得一个Constructor 对象,它代表默认的构造方法,然后调用Constructor对象的newInstance()方法构造一个实例。
 
3)获得对象的所有属性:
Field fields[]=classType.getDeclaredFields();
Class 类的getDeclaredFields()方法返回类的所有属性,包括public、protected、默认和private访问级别的属性
 
(4)获得每个属性相应的getXXX()和setXXX()方法,然后执行这些方法,把原来对象的属性拷贝到新的对象中
 
 
三、用反射机制调用对象的方法
 
public class InvokeTester {
    public int add(int param1, int param2) {
        return param1 + param2;
    }
 
    public String echo(String msg) {
        return "echo: " + msg;
    }
 
    public static void main(String[] args) throws Exception {
        Class<?> classType = InvokeTester.class;
        Object invokeTester = classType.newInstance();
 
        // Object invokeTester = classType.getConstructor(new
        // Class[]{}).newInstance(new Object[]{});
 
 
        //获取InvokeTester类的add()方法
        Method addMethod = classType.getMethod("add", new Class[]{int.class, int.class});
        //调用invokeTester对象上的add()方法
        Object result = addMethod.invoke(invokeTester, new Object[]{new Integer(100), new Integer(200)});
        System.out.println((Integer) result);
 
 
        //获取InvokeTester类的echo()方法
        Method echoMethod = classType.getMethod("echo", new Class[]{String.class});
        //调用invokeTester对象的echo()方法
        result = echoMethod.invoke(invokeTester, new Object[]{"Hello"});
        System.out.println((String) result);
    }
}
 
 
在例程InvokeTester类的main()方法中,运用反射机制调用一个InvokeTester对象的add()和echo()方法
 
add()方法的两个参数为int 类型,获得表示add()方法的Method对象的代码如下:
Method addMethod=classType.getMethod("add",new Class[]{int.class,int.class});
Method类的invoke(Object obj,Object args[])方法接收的参数必须为对象,如果参数为基本类型数据,必须转换为相应的包装类型的对象。invoke()方法的返回值总是对象,如果实际被调用的方法的返回类型是基本类型数据,那么invoke()方法会把它转换为相应的包装类型的对象,再将其返回。
 
在本例中,尽管InvokeTester 类的add()方法的两个参数以及返回值都是int类型,调用add Method 对象的invoke()方法时,只能传递Integer 类型的参数,并且invoke()方法的返回类型也是Integer 类型,Integer 类是int 基本类型的包装类:
 
Object result=addMethod.invoke(invokeTester,
new Object[]{new Integer(100),new Integer(200)});
System.out.println((Integer)result); //result 为Integer类型
 
 
四、动态创建和访问数组
 
java.lang.Array 类提供了动态创建和访问数组元素的各种静态方法。
 
例程ArrayTester1 类的main()方法创建了一个长度为10 的字符串数组,接着把索引位置为5 的元素设为“hello”,然后再读取索引位置为5 的元素的值
 
public class ArrayTester1 {
    public static void main(String args[]) throws Exception {
        Class<?> classType = Class.forName("java.lang.String");
        // 创建一个长度为10的字符串数组
        Object array = Array.newInstance(classType, 10);
        // 把索引位置为5的元素设为"hello"
        Array.set(array, 5, "hello");
        // 获得索引位置为5的元素的值
        String s = (String) Array.get(array, 5);
        System.out.println(s);
    }
}
 
 
例程ArrayTester2 类的main()方法创建了一个 5 x 10 x 15 的整型数组,并把索引位置为[3][5][10] 的元素的值为设37。
 
public class ArrayTester2 {
    public static void main(String args[]) {
        int[] dims = new int[]{5, 10, 15};
        //创建一个具有指定的组件类型和维度的新数组。
        Object array = Array.newInstance(Integer.TYPE, dims);
       
        Object arrayObj = Array.get(array, 3);
        Class<?> cls = arrayObj.getClass().getComponentType();
        System.out.println(cls);
 
        arrayObj = Array.get(arrayObj, 5);
        Array.setInt(arrayObj, 10, 37);
        int arrayCast[][][] = (int[][][]) array;
        System.out.println(arrayCast[3][5][10]);
    }
}
 
 
深入认识Class类
 
众所周知Java有个Object类,是所有Java类的继承根源,其内声明了数个应该在所有Java类中被改写的方法:hashCode()、equals()、clone()、toString()、getClass()等。其中getClass()返回一个Class类的对象。
 
Class类十分特殊。它和一般classes一样继承自Object,其实体用以表达Java程序运行时的classes和interfaces,也用来表达enum、array、primitive Java types
(boolean, byte, char, short, int, long, float, double)以及关键词void。当一个class被加载,或当加载器(class loader)的defineClass()被JVM调用,JVM 便自动产生一个Class object。如果您想借由“修改Java标准库源码”来观察Class object的实际生成时机(例如在Class的constructor内添加一个println()),不能够!因为Class并没有public constructor
 
Class是Reflection起源。针对任何您想探勘的class,唯有先为它产生一个Class object,接下来才能经由后者唤起为数十多个的Reflection APIs
 
Java允许我们从多种途径为一个class生成对应的Class对象。参看本人的《  深入研究java.long.Class类 》一文。
 
欲生成对象实体,在Reflection 动态机制中有两种作法,一个针对“无自变量ctor”,一个针对“带参数ctor”。如果欲调用的是“带参数ctor“就比较麻烦些,不再调用Class的newInstance(),而是调用Constructor 的newInstance()。首先准备一个Class[]做为ctor的参数类型(本例指定
为一个double和一个int),然后以此为自变量调用getConstructor(),获得一个专属ctor。接下来再准备一个Object[] 做为ctor实参值(本例指定3.14159和125),调用上述专属ctor的newInstance()。
 
动态生成“Class object 所对应之class”的对象实体;无自变量。
 
这个动作和上述调用“带参数之ctor”相当类似。首先准备一个Class[]做为参数类型(本例指定其中一个是String,另一个是Hashtable),然后以此为自变量调用getMethod(),获得特定的Method object。接下来准备一个Object[]放置自变量,然后调用上述所得之特定Method object的invoke()。
为什么获得Method object时不需指定回返类型?
 
因为method overloading机制要求signature必须唯一,而回返类型并非signature的一个成份。换句话说,只要指定了method名称和参数列,就一定指出了一个独一无二的method。
 
 
四、运行时变更field内容
 
与先前两个动作相比,“变更field内容”轻松多了,因为它不需要参数和自变量。首先调用Class的getField()并指定field名称。获得特定的Field object之后便可直接调用Field的get()和set()。
 
public class RefFiled {
    public double x;
    public Double y;
 
    public static void main(String args[]) throws NoSuchFieldException, IllegalAccessException {
        Class c = RefFiled.class;
        Field xf = c.getField("x");
        Field yf = c.getField("y");
 
        RefFiled obj = new RefFiled();
 
        System.out.println("变更前x=" + xf.get(obj));
        //变更成员x值
        xf.set(obj, 1.1);
        System.out.println("变更后x=" + xf.get(obj));
 
        System.out.println("变更前y=" + yf.get(obj));
        //变更成员y值
        yf.set(obj, 2.1);
        System.out.println("变更后y=" + yf.get(obj));
    }
}
 
运行结果:
 
变更前x=0.0
变更后x=1.1
变更前y=null
变更后y=2.1
 
Process finished with exit code 0
 
 
参考资料:
此例的部分文字解说和源码来自浪曦论坛 http://bbs.langsin.com。
think in java
Java核心技术
 
声明:本文是在参考了大量资料基础上,摸索运用,总结的基础上完成的。
由于每次书写间隔非常长,参考的资料不能一一写出来,如有侵权,本人将在第一时间删除侵权的内容。
 
 
-----------------
特别说明:
 
Java的反射其实内容远远不至这些,这里看到仅仅是冰山一脚,如果你想要更加深入学习和研究Java的反射机制,你可以参考《Java Reflection in Action》一书,网上有免费下载的。
------------------------------------------------------------------------------------------------------------------------------------原文链接:http://www.blogjava.net/zh-weir/archive/2011/03/26/347063.html

Java反射机制是Java语言被视为准动态语言的关键性质。Java反射机制的核心就是允许在运行时通过Java Reflection APIs来取得已知名字的class类的相关信息,动态地生成此类,并调用其方法或修改其域(甚至是本身声明为private的域或方法)。

也许你使用Java已经很长时间了,可是几乎不会用到Java反射机制。你会嗤之以鼻地告诉我,Java反射机制没啥用。或许在J2EEJ2SE等平台,Java反射机制没啥用(具体我也不了解,不多做评论),但是在Android应用开发中,该机制会带给你许多惊喜。

如果熟悉Android,那么你应该知道,Google不知出于什么原因,在系统源码中一些类或方法中经常加上“@hide”注释标记。它的作用是使这个方法或类在生成SDK时不可见,因此由此注释的东西,你在编译期是不可见的。这就出现了一些问题。一些明明可以访问的东西编译期却无法访问了!这使得你的程序有些本来可以完成的功能无法编译通过。

当然,有一种办法是自己去掉Android源码中的所有“@hide”标记,然后重新编译一份自己的SDK。另一种办法就是使用Java反射机制。当然,你还可以利用反射来访问存在访问限制的方法和修改其域。不过这种使用方法比较特殊,我们在文章的最后单独讨论。

从Class类说起

如果你使用Java,那么你应该知道Java中有一个Class类。Class类本身表示Java对象的类型,我们可以通过一个Object(子)对象的getClass方法取得一个对象的类型,此函数返回的就是一个Class类。当然,获得Class对象的方法有许多,但是没有一种方法是通过Class的构造函数来生成Class对象的。

也许你从来没有使用过Class类,也许你曾以为这是一个没什么用处的东西。不管你以前怎么认为,Class类是整个Java反射机制的源头。一切关于Java反射的故事,都从Class类开始。

因此,要想使用Java反射,我们首先得到Class类的对象。下表列出了几种得到Class类的方法,以供大家参考。

Class object 诞生管道

示例

运用getClass()

注:每个class 都有此函数

String str = "abc";

Class c1 = str.getClass();

运用

Class.getSuperclass()

Button b = new Button();

Class c1 = b.getClass();

Class c2 = c1.getSuperclass();

运用static method

Class.forName()

(最常被使用)

Class c1 = Class.forName ("java.lang.String");

Class c2 = Class.forName ("java.awt.Button");

Class c3 = Class.forName ("java.util.LinkedList$Entry");

Class c4 = Class.forName ("I");

Class c5 = Class.forName ("[I");

运用

.class 语法

Class c1 = String.class;

Class c2 = java.awt.Button.class;

Class c3 = Main.InnerClass.class;

Class c4 = int.class;

Class c5 = int[].class;

运用

primitive wrapper classes

的TYPE 语法

Class c1 = Boolean.TYPE;

Class c2 = Byte.TYPE;

Class c3 = Character.TYPE;

Class c4 = Short.TYPE;

Class c5 = Integer.TYPE;

Class c6 = Long.TYPE;

Class c7 = Float.TYPE;

Class c8 = Double.TYPE;

Class c9 = Void.TYPE;

获取一些基本信息

在我们得到一个类的Class类对象之后,Java反射机制就可以大施拳脚了。首先让我们来了解下如何获取关于某一个类的一些基本信息。

Java class 内部模块

Java class 内部模块说明

相应之Reflection API,多半为Class methods。

返回值类型(return type)

package

class隶属哪个package

getPackage()

Package

import

class导入哪些classes

无直接对应之API。可间接获取。

 

modifier

class(或methods, fields)的属性

int getModifiers()

Modifier.toString(int)

Modifier.isInterface(int)

int

String

bool

class name or interface name

class/interface

名称getName()

String

type parameters

参数化类型的名称

getTypeParameters()

TypeVariable <Class>[]

base class

base class(只可能一个)

getSuperClass()

Class

implemented interfaces

实现有哪些interfaces

getInterfaces()

Class[]

inner classes

内部classes

getDeclaredClasses()

Class[]

outer class

如果我们观察的class 本身是inner classes,那么相对它就会有个outer class。

getDeclaringClass()

Class

上表中,列出了一些Java class内部信息的获取方式。所采用的方法几乎都是调用Class对象的成员方法(由此你就可以了解到Class类的用处了吧)。当然,表中所列出的信息并不是全部,有很大一部分没有列出,你可以通过查阅Java文档得到更全面的了解。另外,下面将重点介绍一下类的构造函数、域和成员方法的获取方式。

类中最重要的三个信息

如果要对一个类的信息重要性进行排名的话,那么这三个信息理应获得前三的名次。它们分别是:构造函数、成员函数、成员变量。

也许你不同意我的排名,没关系。对于Java反射来说,这三个信息与之前介绍的基本信息相比较而言,有着本质的区别。那就是,之前的信息仅仅是只读的,而这三个信息可以在运行时被调用(构造函数和成员函数)或者被修改(成员变量)。所以,我想无可否认,至少站在Java反射机制的立场来说,这三者是最重要的信息。

下面,让我们分别了解一下这三个重要信息的获取方式。另外,我们将在后面的章节,详细介绍他们的调用方式或者修改方式。

构造函数

如果我们将Java对象视为一个二进制的生活在内存中生命体的话,那么构造函数无疑可以类比为Java对象生命体的诞生过程。我们在构造函数调用时为对象分配内存空间,初始化一些属性,于是一个新的生命诞生了。

Java是纯面向对象的语言,Java中几乎所有的一切都是类的对象,因此可想而知构造函数的重要性。

Java反射机制能够得到构造函数信息实在应该是一件令人惊喜的事情。正因为此,反射机制实质上才拥有了孵化生命的能力。换句话言之,我们可以通过反射机制,动态地创建新的对象。

获取构造函数的方法有以下几个:

Constructor getConstructor(Class[] params) 

Constructor[] getConstructors()

Constructor getDeclaredConstructor(Class[] params) 

Constructor[] getDeclaredConstructors()

我们有两种方式对这四个函数分组。

首先可以由构造函数的确定性进行分类。我们知道,一个类实际上可以拥有很多个构造函数。那么我们获取的构造函数是哪个呢?我们可以根据构造函数的参数标签对构造函数进行明确的区分,因此,如果我们在Java反射时指定构造函数的参数,那么我们就能确定地返回我们需要的那个“唯一”的构造函数。getConstructor(Class[] params) getDeclaredConstructor(Class[] params)正是这种确定唯一性的方式。但是,如果我们不清楚每个构造函数的参数表,或者我们出于某种目的需要获取所有的构造函数的信息,那么我们就不需要明确指定参数表,而这时返回的就应该是构造函数数组,因为构造函数很可能不止一个。getConstructors()getDeclaredConstructors()就是这种方式。

另外,我们还可以通过构造函数的访问权限进行分类。在设计类的时候,我们往往有一些构造函数需要声明为“private”、“protect”或者“default”,目的是为了不让外部的类调用此构造函数生成对象。于是,基于访问权限的不同,我们可以将构造函数分为public和非public两种。

getConstructor(Class[] params) getConstructors()仅仅可以获取到public的构造函数,而getDeclaredConstructor(Class[] params) getDeclaredConstructors()则能获取所有(包括public和非public)的构造函数。

成员函数

如果构造函数类比为对象的诞生过程的话,成员函数无疑可以类比为对象的生命行为过程。成员函数的调用执行才是绝大多数对象存在的证据和意义。Java反射机制允许获取成员函数(或者说成员方法)的信息,也就是说,反射机制能够帮助对象践行生命意义。通俗地说,Java反射能使对象完成其相应的功能。

和获取构造函数的方法类似,获取成员函数的方法有以下一些:

Method getMethod(String name, Class[] params)

Method[] getMethods()

Method getDeclaredMethod(String name, Class[] params) 

Method[] getDeclaredMethods() 

其中需要注意,String name参数,需要写入方法名。关于访问权限和确定性的问题,和构造函数基本一致。

成员变量

成员变量,我们经常叫做一个对象的域。从内存的角度来说,构造函数和成员函数都仅仅是Java对象的行为或过程,而成员变量则是真正构成对象本身的细胞和血肉。简单的说,就是成员变量占用的空间之和几乎就是对象占用的所有内存空间。

获取成员变量的方法与上面两种方法类似,具体如下:

Field getField(String name)

Field[] getFields()

Field getDeclaredField(String name)

Field[] getDeclaredFields()

其中,String name参数,需要写入变量名。关于访问权限和确定性的问题,与前面两例基本一致。

让动态真正动起来

在本文的一开始就说,Java反射机制是Java语言被视为准动态语言的关键性质。如果Java反射仅仅能够得到Java类(或对象)运行时的信息,而不能改变其行为和属性,那么它当然算不上“动态”。百度了一把何谓“动态语言”,解释如下:动态语言,是指程序在运行时可以改变其结构:新的函数可以被引进,已有的函数可以被删除等在结构上的变化。由此看来,Java确实不能算作“动态语言”。但是和CC++等纯静态语言相比,Java语言允许使用者在运行时加载、探知、使用编译期间完全未知的classes,所以我们说Java是“准动态”语言。

细心地读者可能已经发现,在“类中最重要的三个信息”一节中,我们获取的信息其实都是属于类的,而不是对象。对于类的信息提取,其实并不涉及到对象内存,在程序编译完成的那一刻起,一切都已经是确定的了。因此,它并不能算“动态”。而如何对对象内存进行操作和访问,才是“动”的真正含义。

说了这么多,关键还在于如何利用反射让Java真正动起来。下面我将按照创生、行为与属性三个方面来介绍反射机制是如何让Java动的。

创生

不知是否本性使然,人类偏爱于思索起源与终结的话题。如果将程序类比于一个二进制的世界的话,那么我们程序员则是这个世界的上帝。我们掌控着这个世界的起源和终结,熟悉世界中一草一木的属性和所有生灵的习性。现在就让我们开始创世纪吧!

在 “构造函数”那一小节中,我们列出了获取构造函数的四种方法。这四种方法的返回值不知是否引起了各位的注意,那就是Constructor类。Constructor就类比于女娲吹给泥人的那一口真气,有了它,一个生命才真正出现。

Constructor支持泛型,也就是它本身应该是Constructor<T>。这个类有一个public成员函数,newInstance(Object... args),其中args为对应的参数。我们正是通过它来实现创生的过程。

行为

行为践行着生命的意义,而众多事物的行为才得以构成整个世界的运转。尽管道家的老子主张“无为而治”,宣扬“圣人处无为之事,行不言之教”,但那是因为他本身就是 “无”的信仰者(“道”即“无”)。我们是唯物主义的信徒,所以必然要以“有”为价值。那么,在二进制的世界里,我们如何调用Java对象的行为呢?

同样,我们首先回顾“成员函数”小节中四种方法的返回值。对,那就是Method类。此类有一个public成员函数,Object invoke(Object receiver, Object... args)。我们能很好理解此函数的第二个参数args,它代表这个方法所需要接收的参数。也许大家对第一个参数receiver还存在疑惑之处。这得从编程语言的发展历程讲起。

如果你关注几种主流编程语言的起源,那么你能有这样的印象:C从汇编而来,C++C而来,而JavaC/C++而来。有这样一种印象就足够了。从这样的发展史我们可以看出,C++Java这两种面向对象的编程语言都是从面向过程的C语言基础上发展而来的。OOP是一种思想,它本身与编程语言无关。也就是说,我们用C也能写出面向对象的程序,这也是C++Java能够以C为基础的根本所在。然而,C无法实现类似object.method()这种表现形式,因为C语言的结构体中并不支持函数定义。那么我们用C实现OOP的时候,如何调用对象的方法呢?

本质上说,object.method()这种调用方式是为了表明具体method()的调用对象。而invoke(Object receiver, Object... args)的第一个参数正是指明调用对象。在C++中,object.method()其实是有隐含参数的,那就是object对象的指针,method原型的第一个参数其实是this指针,于是原型为method(void* this)

这样一溯源,也许你更清楚了Object receiver参数的含义,或许更迷糊了?不管怎样,历史就是如此,只不过我个人能力有限,说不清楚而已。

另外需要注意的是,如果某个方法是Java类的静态方法,那么Object receiver参数可以传入null,因为静态方法不从属于对象。

属性

同样是人类,令狐冲和岳不群是如何被区分开的?那是因为他们有着不同的属性。同样,同一个类可以生成多个对象,几个同类型的对象之间如何区分?属性起着决定性的作用。说到这里,想起一个科幻故事。人体瞬移机,作用的根本原理就是人进入A位置,被完全扫描之后,再在B位置重新组成它的细胞、血肉等属性,从而完全创造出另一个一模一样的人。当然,这是唯物主义的极致,它假设了只要一切物质相同,连记忆和灵魂都不会出现偏差,另外还存在伦理的问题,例如A位置的人会被销毁掉吗?

尽管这是一个科幻,但是在程序的世界里,我们早已经用上了这类似幻想的技术。Java中如何远程传递一个对象?我们已经使用上了Java对象序列化的接口。不仅如此,利用序列化接口,我们甚至可以将一个生命保存起来,在需要的时候将它复活,这就是对象的持久化。不得不感慨,在程序的世界里,我们就是上帝啊!

对象序列化如此强大,那么它的本质是什么呢?它的工作原理是怎样的呢?简单的说,对象序列化的本质就是属性的序列化。原理就是我们崇尚的唯物主义,如果同一个类的两个对象所有属性值都完全相同,那么我们可以认为这是同一个对象。

说了这么多,只是想说明一件事情,属性对于对象而言是多么的重要。那么如何读写对象中属性的值呢?回顾获取属性信息的方法返回值类型,那是FieldField类有两个public方法,分别对应读与写,它们是:

Object get(Object object)

void set(Object object, Object value)

object参数需要传入的对象,原理类似于成员方法需要指明对象一样。如果是静态属性,此值同样可以为null

关于反射的一些高级话题

如果说前面那些属于Java反射的基本知识,那么在文章的最后,我们来探讨一下反射的一些高级话题。另外,本文对基础知识的讲解仅属于抓主干,具体的一些旁支可以自己参看文档。需要提一下的是,Java反射中对数组做过单独的优化处理,具体可查看java.lang.reflect.Array类;还有关于泛型的支持,可查看java.lang.reflect.ParameterizedType及相关资料。

暂时想到的高级话题有三个,由于对Java反射理解的也不算深入,所以仅仅从思路上进行探讨,具体实现上,大家可以参考其他相关资料,做更深入研究。

Android编译期问题

Android的安全权限问题我把它简单的划分成三个层次,最不严格的一层就是仅仅骗过编译器的“@hide”标记。对于一款开源的操作系统而言,这个标记本身并不具备安全上的限制。不过,从上次Google过来的负责Android工程师的说法来看,这个标记的作用更多的是方便硬件厂商做闭源的二次开发。这样解释倒也说得过去。

不过这并不影响我们使用反射机制以绕过原生Android的第一层安全措施。如果你熟悉源码的话,会发现这可以应用到很多地方。并且最关键的是你并不需要放在源码中编译,而是像普通应用程序的开发过程一样。

具体使用范围我不能一一列举了,例如自定义窗口、安装程序等等。简单的说,在Android上使用反射技术,你才会对Android系统有更深的理解和更高的控制权。

软件的解耦合

我们在架构代码的时候,经常提到解耦合、弱耦合。其实,解耦和不仅仅只能在代码上做文章。我们可以考虑这样一种情况:软件的功能需求不可能一开始就完全确定,有一些功能在软件开发的后期甚至是软件已经发布出去之后才想到要加入或者去掉。

按我们惯有的思维,这种情况就得改动源码,重新编译。如果软件已经发布出去,那么就得让客户重新安装一次软件。反思一下,我们是否认为软件和程序是同一回事呢?事实上,如果你能将软件和程序分开来理解,那么你会发现,为了应对以上的情况,我们还有其他的解决办法。

我国有一个很重要但是很麻烦的制度,那就是户籍制度。它的本意是为了更好的管理人口事宜。每当一个孩子出生,我们就需要在户籍管理的地方去给他办理户籍入户;而每当一个人去世,我们也需要在相应的地方销去他的户籍。既然我们可以视类为生命,那么我们能否通过学习这样的户籍管理制度来动态地管理类呢?

事实上这样的管理是可行的,而且Java虚拟机本身正是基于这样的机制来运行程序的。因此我们是否可以这样来架构软件框架。首先,我们的软件有一个配置文件,配置文件其实是一个文本,里面详细描述了,我们的软件核心部分运行起来后还需要从什么路径加载些什么类需要何时调用什么方法等。这样当我们需要加或减某些功能时,我们只需要简单地修改配置文本文件,然后删除或者添加相应的.class文件就可以了。

如果你足够敏感,你或许会发现,这种方式形成的配置文件几乎可以相当于一门脚本语言了。而且这个脚本的解释器也是我们自己写的,另外关键是它是开发的,你可以为它动态地加入一些新的类以增加它的功能。

不要以为这仅仅是一个设想,虽然要开发成一门完备的脚本语言确实比较麻烦。但是在一些网络端的大型项目中,通过配置文件 + ClassLoader + 反射机制结合形成的这种软件解耦和方式已经用得比较普遍了。

所以,在此我不是在提出一种设想,而是在介绍业界处理此类问题的一种解决方案。

反射安全

文章读到这里,我想你应该由衷地感叹,Java反射机制实在是太强大了。但是,如果你有一些安全意识的话,就会发现Java这个机制强大得似乎有些过头了。前面我们提到,Java反射甚至可以访问private方法和属性。为了让大家对Java反射有更全面的了解,树立正确的人生观价值观,本小节将对Java的安全问题做一个概要性的介绍。

相对于C++来说,Java算是比较安全的语言了。这与它们的运行机制有密切的关系,C++运行于本地,也就是说几乎所有程序的权限理论上都是相同的。而Java由于是运行于虚拟机中,而不直接与外部联系,所以实际上Java的运行环境是一个“沙盒”环境。

Java的安全机制其实是比较复杂的,至少对于我来说是如此。作为Java的安全模型,它包括了:字节码验证器、类加载器、安全管理器、访问控制器等一系列的组件。之前文中提到过,我把Android安全权限划分为三个等级:第一级是针对编译期的“@hide”标记;第二级是针对访问权限的private等修饰;第三级则是以安全管理器为托管的Permission机制。

Java反射确实可以访问private的方法和属性,这是绕过第二级安全机制的方法(之一)。它其实是Java本身为了某种目的而留下的类似于“后门”的东西,或者说是为了方便调试?不管如何,它的原理其实是关闭访问安全检查。

如果你具有独立钻研的精神的话,你会发现之前我们提到的FieldMethodConstructor类,它们都有一个共同的父类AccessibleObject AccessibleObject 有一个公共方法:void setAccessible(boolean flag)。正是这个方法,让我们可以改变动态的打开或者关闭访问安全检查,从而访问到原本是private的方法或域。另外,访问安全检查是一件比较耗时的操作,关闭它反射的性能也会有较大提升。

不要认为我们绕过了前两级安全机制就沾沾自喜了,因为这两级安全并不是真正为了安全而设置的。它们的作用更多的是为了更好的完善规则。而第三级安全才是真正为了防止恶意攻击而出现的。在这一级的防护下,你甚至可能都无法完成反射(ReflectPermission),其他的一切自然无从说起。

对于这一级,我的了解还太少,并且也与本文的主题相关甚少。以后有机会,深入学习之后再聊吧!

参考文献

转 侯捷谈Java反射机制

java反射机制

java反射的若干高级应用

Java反射经典实例 Java Reflection Cookbook

Java ClassLoader详解》

《自定义ClassLoader的简单例子》

《提高Java反射速度的方法以及对setAccessable的误解》

《深入理解Java安全模型》

Java安全管理器(Security Manager)

------------------------------------------------------------------------------------------------------------------------------------原文链接:http://www.cnblogs.com/rollenholt/archive/2011/09/02/2163758.html

本篇文章依旧采用小例子来说明,因为我始终觉的,案例驱动是最好的,要不然只看理论的话,看了也不懂,不过建议大家在看完文章之后,在回过头去看看理论,会有更好的理解。

下面开始正文。

【案例1】通过一个对象获得完整的包名和类名

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
package Reflect;
 
/**
  * 通过一个对象获得完整的包名和类名
  * */
class Demo{
     //other codes...
}
 
class hello{
     public static void main(String[] args) {
         Demo demo= new Demo();
         System.out.println(demo.getClass().getName());
     }
}

【运行结果】:Reflect.Demo

添加一句:所有类的对象其实都是Class的实例。

【案例2】实例化Class类对象

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
package Reflect;
class Demo{
     //other codes...
}
 
class hello{
     public static void main(String[] args) {
         Class<?> demo1= null ;
         Class<?> demo2= null ;
         Class<?> demo3= null ;
         try {
             //一般尽量采用这种形式
             demo1=Class.forName( "Reflect.Demo" );
         } catch (Exception e){
             e.printStackTrace();
         }
         demo2= new Demo().getClass();
         demo3=Demo. class ;
         
         System.out.println( "类名称   " +demo1.getName());
         System.out.println( "类名称   " +demo2.getName());
         System.out.println( "类名称   " +demo3.getName());
         
     }
}

【运行结果】:

类名称   Reflect.Demo

类名称   Reflect.Demo

类名称   Reflect.Demo

【案例3】通过Class实例化其他类的对象

通过无参构造实例化对象

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
package Reflect;
 
class Person{
     
     public String getName() {
         return name;
     }
     public void setName(String name) {
         this .name = name;
     }
     public int getAge() {
         return age;
     }
     public void setAge( int age) {
         this .age = age;
     }
     @Override
     public String toString(){
         return "[" + this .name+ "  " + this .age+ "]" ;
     }
     private String name;
     private int age;
}
 
class hello{
     public static void main(String[] args) {
         Class<?> demo= null ;
         try {
             demo=Class.forName( "Reflect.Person" );
         } catch (Exception e) {
             e.printStackTrace();
         }
         Person per= null ;
         try {
             per=(Person)demo.newInstance();
         } catch (InstantiationException e) {
             // TODO Auto-generated catch block
             e.printStackTrace();
         } catch (IllegalAccessException e) {
             // TODO Auto-generated catch block
             e.printStackTrace();
         }
         per.setName( "Rollen" );
         per.setAge( 20 );
         System.out.println(per);
     }
}

【运行结果】:

[Rollen  20]

但是注意一下,当我们把Person中的默认的无参构造函数取消的时候,比如自己定义只定义一个有参数的构造函数之后,会出现错误:

比如我定义了一个构造函数:

?
1
2
3
4
public Person(String name, int age) {
         this .age=age;
         this .name=name;
     }

然后继续运行上面的程序,会出现:

java.lang.InstantiationException: Reflect.Person

    at java.lang.Class.newInstance0(Class.java:340)

    at java.lang.Class.newInstance(Class.java:308)

    at Reflect.hello.main(hello.java:39)

Exception in thread "main" java.lang.NullPointerException

    at Reflect.hello.main(hello.java:47)

所以大家以后再编写使用Class实例化其他类的对象的时候,一定要自己定义无参的构造函数

 

【案例】通过Class调用其他类中的构造函数 (也可以通过这种方式通过Class创建其他类的对象)

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
package Reflect;
 
import java.lang.reflect.Constructor;
 
class Person{
     
     public Person() {
         
     }
     public Person(String name){
         this .name=name;
     }
     public Person( int age){
         this .age=age;
     }
     public Person(String name, int age) {
         this .age=age;
         this .name=name;
     }
     public String getName() {
         return name;
     }
     public int getAge() {
         return age;
     }
     @Override
     public String toString(){
         return "[" + this .name+ "  " + this .age+ "]" ;
     }
     private String name;
     private int age;
}
 
class hello{
     public static void main(String[] args) {
         Class<?> demo= null ;
         try {
             demo=Class.forName( "Reflect.Person" );
         } catch (Exception e) {
             e.printStackTrace();
         }
         Person per1= null ;
         Person per2= null ;
         Person per3= null ;
         Person per4= null ;
         //取得全部的构造函数
         Constructor<?> cons[]=demo.getConstructors();
         try {
             per1=(Person)cons[ 0 ].newInstance();
             per2=(Person)cons[ 1 ].newInstance( "Rollen" );
             per3=(Person)cons[ 2 ].newInstance( 20 );
             per4=(Person)cons[ 3 ].newInstance( "Rollen" , 20 );
         } catch (Exception e){
             e.printStackTrace();
         }
         System.out.println(per1);
         System.out.println(per2);
         System.out.println(per3);
         System.out.println(per4);
     }
}

【运行结果】:

[null  0]

[Rollen  0]

[null  20]

[Rollen  20]

【案例】 

返回一个类实现的接口:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
package Reflect;
 
interface China{
     public static final String name= "Rollen" ;
     public static  int age= 20 ;
     public void sayChina();
     public void sayHello(String name, int age);
}
 
class Person implements China{
     public Person() {
         
     }
     public Person(String sex){
         this .sex=sex;
     }
     public String getSex() {
         return sex;
     }
     public void setSex(String sex) {
         this .sex = sex;
     }
     @Override
     public void sayChina(){
         System.out.println( "hello ,china" );
     }
     @Override
     public void sayHello(String name, int age){
         System.out.println(name+ "  " +age);
     }
     private String sex;
}
 
class hello{
     public static void main(String[] args) {
         Class<?> demo= null ;
         try {
             demo=Class.forName( "Reflect.Person" );
         } catch (Exception e) {
             e.printStackTrace();
         }
         //保存所有的接口
         Class<?> intes[]=demo.getInterfaces();
         for ( int i = 0 ; i < intes.length; i++) {
             System.out.println( "实现的接口   " +intes[i].getName());
         }
     }
}

【运行结果】:

实现的接口   Reflect.China

(注意,以下几个例子,都会用到这个例子的Person类,所以为节省篇幅,此处不再粘贴Person的代码部分,只粘贴主类hello的代码)

【案例】:取得其他类中的父类

?
1
2
3
4
5
6
7
8
9
10
11
12
13
class hello{
     public static void main(String[] args) {
         Class<?> demo= null ;
         try {
             demo=Class.forName( "Reflect.Person" );
         } catch (Exception e) {
             e.printStackTrace();
         }
         //取得父类
         Class<?> temp=demo.getSuperclass();
         System.out.println( "继承的父类为:   " +temp.getName());
     }
}

【运行结果】

继承的父类为:   java.lang.Object

【案例】:获得其他类中的全部构造函数

这个例子需要在程序开头添加import java.lang.reflect.*;

然后将主类编写为:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
class hello{
     public static void main(String[] args) {
         Class<?> demo= null ;
         try {
             demo=Class.forName( "Reflect.Person" );
         } catch (Exception e) {
             e.printStackTrace();
         }
         Constructor<?>cons[]=demo.getConstructors();
         for ( int i = 0 ; i < cons.length; i++) {
             System.out.println( "构造方法:  " +cons[i]);
         }
     }
}

【运行结果】:

构造方法:  public Reflect.Person()

构造方法:  public Reflect.Person(java.lang.String)

但是细心的读者会发现,上面的构造函数没有public 或者private这一类的修饰符

下面这个例子我们就来获取修饰符

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
class hello{
     public static void main(String[] args) {
         Class<?> demo= null ;
         try {
             demo=Class.forName( "Reflect.Person" );
         } catch (Exception e) {
             e.printStackTrace();
         }
         Constructor<?>cons[]=demo.getConstructors();
         for ( int i = 0 ; i < cons.length; i++) {
             Class<?> p[]=cons[i].getParameterTypes();
             System.out.print( "构造方法:  " );
             int mo=cons[i].getModifiers();
             System.out.print(Modifier.toString(mo)+ " " );
             System.out.print(cons[i].getName());
             System.out.print( "(" );
             for ( int j= 0 ;j<p.length;++j){
                 System.out.print(p[j].getName()+ " arg" +i);
                 if (j<p.length- 1 ){
                     System.out.print( "," );
                 }
             }
             System.out.println( "){}" );
         }
     }
}

【运行结果】:

构造方法:  public Reflect.Person(){}

构造方法:  public Reflect.Person(java.lang.String arg1){}

有时候一个方法可能还有异常,呵呵。下面看看:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
class hello{
     public static void main(String[] args) {
         Class<?> demo= null ;
         try {
             demo=Class.forName( "Reflect.Person" );
         } catch (Exception e) {
             e.printStackTrace();
         }
         Method method[]=demo.getMethods();
         for ( int i= 0 ;i<method.length;++i){
             Class<?> returnType=method[i].getReturnType();
             Class<?> para[]=method[i].getParameterTypes();
             int temp=method[i].getModifiers();
             System.out.print(Modifier.toString(temp)+ " " );
             System.out.print(returnType.getName()+ "  " );
             System.out.print(method[i].getName()+ " " );
             System.out.print( "(" );
             for ( int j= 0 ;j<para.length;++j){
                 System.out.print(para[j].getName()+ " " + "arg" +j);
                 if (j<para.length- 1 ){
                     System.out.print( "," );
                 }
             }
             Class<?> exce[]=method[i].getExceptionTypes();
             if (exce.length> 0 ){
                 System.out.print( ") throws " );
                 for ( int k= 0 ;k<exce.length;++k){
                     System.out.print(exce[k].getName()+ " " );
                     if (k<exce.length- 1 ){
                         System.out.print( "," );
                     }
                 }
             } else {
                 System.out.print( ")" );
             }
             System.out.println();
         }
     }
}

【运行结果】:

public java.lang.String  getSex ()

public void  setSex (java.lang.String arg0)

public void  sayChina ()

public void  sayHello (java.lang.String arg0,int arg1)

public final native void  wait (long arg0) throws java.lang.InterruptedException

public final void  wait () throws java.lang.InterruptedException

public final void  wait (long arg0,int arg1) throws java.lang.InterruptedException

public boolean  equals (java.lang.Object arg0)

public java.lang.String  toString ()

public native int  hashCode ()

public final native java.lang.Class  getClass ()

public final native void  notify ()

public final native void  notifyAll ()

【案例】接下来让我们取得其他类的全部属性吧,最后我讲这些整理在一起,也就是通过class取得一个类的全部框架

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
class hello {
     public static void main(String[] args) {
         Class<?> demo = null ;
         try {
             demo = Class.forName( "Reflect.Person" );
         } catch (Exception e) {
             e.printStackTrace();
         }
         System.out.println( "===============本类属性========================" );
         // 取得本类的全部属性
         Field[] field = demo.getDeclaredFields();
         for ( int i = 0 ; i < field.length; i++) {
             // 权限修饰符
             int mo = field[i].getModifiers();
             String priv = Modifier.toString(mo);
             // 属性类型
             Class<?> type = field[i].getType();
             System.out.println(priv + " " + type.getName() + " "
                     + field[i].getName() + ";" );
         }
         System.out.println( "===============实现的接口或者父类的属性========================" );
         // 取得实现的接口或者父类的属性
         Field[] filed1 = demo.getFields();
         for ( int j = 0 ; j < filed1.length; j++) {
             // 权限修饰符
             int mo = filed1[j].getModifiers();
             String priv = Modifier.toString(mo);
             // 属性类型
             Class<?> type = filed1[j].getType();
             System.out.println(priv + " " + type.getName() + " "
                     + filed1[j].getName() + ";" );
         }
     }
}

【运行结果】:

===============本类属性========================

private java.lang.String sex;

===============实现的接口或者父类的属性========================

public static final java.lang.String name;

public static final int age;

【案例】其实还可以通过反射调用其他类中的方法:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
class hello {
     public static void main(String[] args) {
         Class<?> demo = null ;
         try {
             demo = Class.forName( "Reflect.Person" );
         } catch (Exception e) {
             e.printStackTrace();
         }
         try {
             //调用Person类中的sayChina方法
             Method method=demo.getMethod( "sayChina" );
             method.invoke(demo.newInstance());
             //调用Person的sayHello方法
             method=demo.getMethod( "sayHello" , String. class , int . class );
             method.invoke(demo.newInstance(), "Rollen" , 20 );
             
         } catch (Exception e) {
             e.printStackTrace();
         }
     }
}

  【运行结果】:

hello ,china

Rollen  20

【案例】调用其他类的set和get方法

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
class hello {
     public static void main(String[] args) {
         Class<?> demo = null ;
         Object obj= null ;
         try {
             demo = Class.forName( "Reflect.Person" );
         } catch (Exception e) {
             e.printStackTrace();
         }
         try {
          obj=demo.newInstance();
         } catch (Exception e) {
             e.printStackTrace();
         }
         setter(obj, "Sex" , "男" ,String. class );
         getter(obj, "Sex" );
     }
 
     /**
      * @param obj
      *            操作的对象
      * @param att
      *            操作的属性
      * */
     public static void getter(Object obj, String att) {
         try {
             Method method = obj.getClass().getMethod( "get" + att);
             System.out.println(method.invoke(obj));
         } catch (Exception e) {
             e.printStackTrace();
         }
     }
 
     /**
      * @param obj
      *            操作的对象
      * @param att
      *            操作的属性
      * @param value
      *            设置的值
      * @param type
      *            参数的属性
      * */
     public static void setter(Object obj, String att, Object value,
             Class<?> type) {
         try {
             Method method = obj.getClass().getMethod( "set" + att, type);
             method.invoke(obj, value);
         } catch (Exception e) {
             e.printStackTrace();
         }
     }
} // end class

【运行结果】:

 【案例】通过反射操作属性

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
class hello {
     public static void main(String[] args) throws Exception {
         Class<?> demo = null ;
         Object obj = null ;
 
         demo = Class.forName( "Reflect.Person" );
         obj = demo.newInstance();
 
         Field field = demo.getDeclaredField( "sex" );
         field.setAccessible( true );
         field.set(obj, "男" );
         System.out.println(field.get(obj));
     }
} // end class

【案例】通过反射取得并修改数组的信息:

?
1
2
3
4
5
6
7
8
9
10
11
12
import java.lang.reflect.*;
class hello{
     public static void main(String[] args) {
         int [] temp={ 1 , 2 , 3 , 4 , 5 };
         Class<?>demo=temp.getClass().getComponentType();
         System.out.println( "数组类型: " +demo.getName());
         System.out.println( "数组长度  " +Array.getLength(temp));
         System.out.println( "数组的第一个元素: " +Array.get(temp, 0 ));
         Array.set(temp, 0 , 100 );
         System.out.println( "修改之后数组第一个元素为: " +Array.get(temp, 0 ));
     }
}

【运行结果】:

数组类型: int

数组长度  5

数组的第一个元素: 1

修改之后数组第一个元素为: 100

【案例】通过反射修改数组大小

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
class hello{
     public static void main(String[] args) {
         int [] temp={ 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9 };
         int [] newTemp=( int [])arrayInc(temp, 15 );
         print(newTemp);
         System.out.println( "=====================" );
         String[] atr={ "a" , "b" , "c" };
         String[] str1=(String[])arrayInc(atr, 8 );
         print(str1);
     }
     
     /**
      * 修改数组大小
      * */
     public static Object arrayInc(Object obj, int len){
         Class<?>arr=obj.getClass().getComponentType();
         Object newArr=Array.newInstance(arr, len);
         int co=Array.getLength(obj);
         System.arraycopy(obj, 0 , newArr, 0 , co);
         return newArr;
     }
     /**
      * 打印
      * */
     public static void print(Object obj){
         Class<?>c=obj.getClass();
         if (!c.isArray()){
             return ;
         }
         System.out.println( "数组长度为: " +Array.getLength(obj));
         for ( int i = 0 ; i < Array.getLength(obj); i++) {
             System.out.print(Array.get(obj, i)+ " " );
         }
     }
}

【运行结果】:

数组长度为: 15

1 2 3 4 5 6 7 8 9 0 0 0 0 0 0 =====================

数组长度为: 8

a b c null null null null null

动态代理

【案例】首先来看看如何获得类加载器:

?
1
2
3
4
5
6
7
8
9
class test{
     
}
class hello{
     public static void main(String[] args) {
         test t= new test();
         System.out.println( "类加载器  " +t.getClass().getClassLoader().getClass().getName());
     }
}

【程序输出】:

类加载器  sun.misc.Launcher$AppClassLoader

其实在java中有三种类类加载器。

1)Bootstrap ClassLoader 此加载器采用c++编写,一般开发中很少见。

2)Extension ClassLoader 用来进行扩展类的加载,一般对应的是jre\lib\ext目录中的类

3)AppClassLoader 加载classpath指定的类,是最常用的加载器。同时也是java中默认的加载器。

如果想要完成动态代理,首先需要定义一个InvocationHandler接口的子类,已完成代理的具体操作。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
package Reflect;
import java.lang.reflect.*;
 
//定义项目接口
interface Subject {
     public String say(String name, int age);
}
 
// 定义真实项目
class RealSubject implements Subject {
     @Override
     public String say(String name, int age) {
         return name + "  " + age;
     }
}
 
class MyInvocationHandler implements InvocationHandler {
     private Object obj = null ;
 
     public Object bind(Object obj) {
         this .obj = obj;
         return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj
                 .getClass().getInterfaces(), this );
     }
 
     @Override
     public Object invoke(Object proxy, Method method, Object[] args)
             throws Throwable {
         Object temp = method.invoke( this .obj, args);
         return temp;
     }
}
 
class hello {
     public static void main(String[] args) {
         MyInvocationHandler demo = new MyInvocationHandler();
         Subject sub = (Subject) demo.bind( new RealSubject());
         String info = sub.say( "Rollen" , 20 );
         System.out.println(info);
     }
}

【运行结果】:

Rollen  20

类的生命周期

在一个类编译完成之后,下一步就需要开始使用类,如果要使用一个类,肯定离不开JVM。在程序执行中JVM通过装载,链接,初始化这3个步骤完成。

类的装载是通过类加载器完成的,加载器将.class文件的二进制文件装入JVM的方法区,并且在堆区创建描述这个类的java.lang.Class对象。用来封装数据。但是同一个类只会被类装载器装载以前

链接就是把二进制数据组装为可以运行的状态。

 

链接分为校验,准备,解析这3个阶段

校验一般用来确认此二进制文件是否适合当前的JVM(版本),

准备就是为静态成员分配内存空间,。并设置默认值

解析指的是转换常量池中的代码作为直接引用的过程,直到所有的符号引用都可以被运行程序使用(建立完整的对应关系)

完成之后,类型也就完成了初始化,初始化之后类的对象就可以正常使用了,直到一个对象不再使用之后,将被垃圾回收。释放空间。

当没有任何引用指向Class对象时就会被卸载,结束类的生命周期

将反射用于工厂模式

先来看看,如果不用反射的时候,的工厂模式吧:

http://www.cnblogs.com/rollenholt/archive/2011/08/18/2144851.html

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
/**
  * @author Rollen-Holt 设计模式之 工厂模式
  */
 
interface fruit{
     public abstract void eat();
}
 
class Apple implements fruit{
     public void eat(){
         System.out.println( "Apple" );
     }
}
 
class Orange implements fruit{
     public void eat(){
         System.out.println( "Orange" );
     }
}
 
// 构造工厂类
// 也就是说以后如果我们在添加其他的实例的时候只需要修改工厂类就行了
class Factory{
     public static fruit getInstance(String fruitName){
         fruit f= null ;
         if ( "Apple" .equals(fruitName)){
             f= new Apple();
         }
         if ( "Orange" .equals(fruitName)){
             f= new Orange();
         }
         return f;
     }
}
class hello{
     public static void main(String[] a){
         fruit f=Factory.getInstance( "Orange" );
         f.eat();
     }
 
}

这样,当我们在添加一个子类的时候,就需要修改工厂类了。如果我们添加太多的子类的时候,改的就会很多。

现在我们看看利用反射机制:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
package Reflect;
 
interface fruit{
     public abstract void eat();
}
 
class Apple implements fruit{
     public void eat(){
         System.out.println( "Apple" );
     }
}
 
class Orange implements fruit{
     public void eat(){
         System.out.println( "Orange" );
     }
}
 
class Factory{
     public static fruit getInstance(String ClassName){
         fruit f= null ;
         try {
             f=(fruit)Class.forName(ClassName).newInstance();
         } catch (Exception e) {
             e.printStackTrace();
         }
         return f;
     }
}
class hello{
     public static void main(String[] a){
         fruit f=Factory.getInstance( "Reflect.Apple" );
         if (f!= null ){
             f.eat();
         }
     }
}

现在就算我们添加任意多个子类的时候,工厂类就不需要修改。

 

上面的爱吗虽然可以通过反射取得接口的实例,但是需要传入完整的包和类名。而且用户也无法知道一个接口有多少个可以使用的子类,所以我们通过属性文件的形式配置所需要的子类。

下面我们来看看: 结合属性文件的工厂模式

首先创建一个fruit.properties的资源文件,

内容为:

?
1
2
apple=Reflect.Apple
orange=Reflect.Orange

 然后编写主类代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
package Reflect;
 
import java.io.*;
import java.util.*;
 
interface fruit{
     public abstract void eat();
}
 
class Apple implements fruit{
     public void eat(){
         System.out.println( "Apple" );
     }
}
 
class Orange implements fruit{
     public void eat(){
         System.out.println( "Orange" );
     }
}
 
//操作属性文件类
class init{
     public static Properties getPro() throws FileNotFoundException, IOException{
         Properties pro= new Properties();
         File f= new File( "fruit.properties" );
         if (f.exists()){
             pro.load( new FileInputStream(f));
         } else {
             pro.setProperty( "apple" , "Reflect.Apple" );
             pro.setProperty( "orange" , "Reflect.Orange" );
             pro.store( new FileOutputStream(f), "FRUIT CLASS" );
         }
         return pro;
     }
}
 
class Factory{
     public static fruit getInstance(String ClassName){
         fruit f= null ;
         try {
             f=(fruit)Class.forName(ClassName).newInstance();
         } catch (Exception e) {
             e.printStackTrace();
         }
         return f;
     }
}
class hello{
     public static void main(String[] a) throws FileNotFoundException, IOException{
         Properties pro=init.getPro();
         fruit f=Factory.getInstance(pro.getProperty( "apple" ));
         if (f!= null ){
             f.eat();
         }
     }
}

【运行结果】:Apple

------------------------------------------------------------------------------------------------------------------------------------原文链接:http://azrael6619.iteye.com/blog/429797

AVA反射机制

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
Java反射机制主要提供了以下功能: 在运行时判断任意一个对象所属的类;在运行时构造任意一个类的对象;在运行时判断任意一个类所具有的成员变量和方法;在运行时调用任意一个对象的方法;生成动态代理。
1. 得到某个对象的属性

Java代码   收藏代码
  1. public Object getProperty(Object owner, String fieldName) throws Exception {  
  2.      Class ownerClass = owner.getClass();  
  3.    
  4.      Field field = ownerClass.getField(fieldName);  
  5.    
  6.      Object property = field.get(owner);  
  7.    
  8.      return property;  
  9. }  

 
Class ownerClass = owner.getClass():得到该对象的Class。

Field field = ownerClass.getField(fieldName):通过Class得到类声明的属性。

Object property = field.get(owner):通过对象得到该属性的实例,如果这个属性是非公有的,这里会报IllegalAccessException。

2. 得到某个类的静态属性

Java代码   收藏代码
  1. public Object getStaticProperty(String className, String fieldName)  
  2.              throws Exception {  
  3.      Class ownerClass = Class.forName(className);  
  4.    
  5.      Field field = ownerClass.getField(fieldName);  
  6.    
  7.      Object property = field.get(ownerClass);  
  8.    
  9.      return property;  
  10. }  

 

Class ownerClass = Class.forName(className) :首先得到这个类的Class。

Field field = ownerClass.getField(fieldName):和上面一样,通过Class得到类声明的属性。

Object property = field.get(ownerClass) :这里和上面有些不同,因为该属性是静态的,所以直接从类的Class里取。

3. 执行某对象的方法

Java代码   收藏代码
  1. public Object invokeMethod(Object owner, String methodName, Object[] args) throws Exception {  
  2.    
  3.      Class ownerClass = owner.getClass();  
  4.    
  5.      Class[] argsClass = new Class[args.length];  
  6.    
  7.      for (int i = 0, j = args.length; i < j; i++) {  
  8.          argsClass[i] = args[i].getClass();  
  9.      }  
  10.   
  11.       Method method = ownerClass.getMethod(methodName,argsClass);  
  12.    
  13.      return method.invoke(owner, args);  
  14. }  

 
Class owner_class = owner.getClass() :首先还是必须得到这个对象的Class。

5~9行:配置参数的Class数组,作为寻找Method的条件。

Method method = ownerClass.getMethod(methodName, argsClass):通过methodName和参数的argsClass(方法中的参数类型集合)数组得到要执行的Method。

method.invoke(owner, args):执行该Method.invoke方法的参数是执行这个方法的对象owner,和参数数组args,可以这么理解:owner对象中带有参数args的method方法。返回值是Object,也既是该方法的返回值。

4. 执行某个类的静态方法

Java代码   收藏代码
  1. public Object invokeStaticMethod(String className, String methodName,  
  2.              Object[] args) throws Exception {  
  3.      Class ownerClass = Class.forName(className);  
  4.    
  5.      Class[] argsClass = new Class[args.length];  
  6.    
  7.      for (int i = 0, j = args.length; i < j; i++) {  
  8.          argsClass[i] = args[i].getClass();  
  9.      }  
  10.    
  11.     Method method = ownerClass.getMethod(methodName,argsClass);  
  12.    
  13.      return method.invoke(null, args);  
  14.  }  

 

基本的原理和实例3相同,不同点是最后一行,invoke的一个参数是null,因为这是静态方法,不需要借助实例运行。

5. 新建实例

Java代码   收藏代码
  1. public Object newInstance(String className, Object[] args) throws Exception {  
  2.      Class newoneClass = Class.forName(className);  
  3.    
  4.      Class[] argsClass = new Class[args.length];  
  5.    
  6.      for (int i = 0, j = args.length; i < j; i++) {  
  7.          argsClass[i] = args[i].getClass();  
  8.      }  
  9.    
  10.      Constructor cons = newoneClass.getConstructor(argsClass);  
  11.    
  12.      return cons.newInstance(args);  
  13.    
  14. }  

 
这里说的方法是执行带参数的构造函数来新建实例的方法。如果不需要参数,可以直接使用newoneClass.newInstance()来实现。

Class newoneClass = Class.forName(className):第一步,得到要构造的实例的Class。

第5~第9行:得到参数的Class数组。

Constructor cons = newoneClass.getConstructor(argsClass):得到构造子。

cons.newInstance(args):新建实例。

6. 判断是否为某个类的实例

Java代码   收藏代码
  1. public boolean isInstance(Object obj, Class cls) {  
  2.      return cls.isInstance(obj);  
  3. }  

 
7. 得到数组中的某个元素

Java代码   收藏代码
  1. public Object getByArray(Object array, int index) {  
  2.      return Array.get(array,index);  
  3. }  


  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值