java反射机制

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

1.  java.lang.Class对象

2. 代码举例:

  
  
  
  
  1. import java.lang.reflect.Field;  
  2. import java.lang.reflect.Method;  
  3. import java.lang.reflect.Modifier;  
  4.  
  5. public class ReflectionTest {  
  6.     public static void main(String[] args) {  
  7.         Class c = null;  
  8.         try {  
  9.             c = Class.forName("java.lang.String");  
  10.             System.out.println("package " + c.getPackage().getName() + ";");  
  11.             System.out.print(Modifier.toString(c.getModifiers()) + " ");  
  12.             System.out.print("class " + c.getSimpleName() + " ");  
  13.             if (c.getSuperclass() != Object.class) {  
  14.                 System.out.print("extends " + c.getSuperclass().getSimpleName());  
  15.             }  
  16.             Class[] inters = c.getInterfaces();  
  17.             if (inters.length > 0) {  
  18.                 System.out.print("implements ");  
  19.                 for (int i = 0; i < inters.length; i++) {  
  20.                     System.out.print(inters[i].getSimpleName());  
  21.                     if (i < inters.length - 1) {  
  22.                         System.out.print(",");  
  23.                     }  
  24.                 }  
  25.             }  
  26.             System.out.println("{");  
  27.             printFields(c);  
  28.             printMethods(c);  
  29.             System.out.println("}");  
  30.         } catch (ClassNotFoundException e) {  
  31.             // TODO Auto-generated catch block  
  32.             e.printStackTrace();  
  33.         }  
  34.     }  
  35.  
  36.     public static void printFields(Class c) {  
  37.         Field[] field = c.getDeclaredFields();  
  38.         if (field.length > 0) {  
  39.             for (int i = 0; i < field.length; i++) {  
  40.                 System.out.println(Modifier.toString(field[i].getModifiers()) + " " + field[i].getType().getSimpleName() + " " + field[i].getName() + ";");  
  41.             }  
  42.         }  
  43.     }  
  44.  
  45.     public static void printMethods(Class c) {  
  46.         Method[] method = c.getDeclaredMethods();  
  47.         if (method.length > 0) {  
  48.             for (int i = 0; i < method.length; i++) {  
  49.                 Class[] parameter = method[i].getParameterTypes();  
  50.                 System.out.print(Modifier.toString(method[i].getModifiers())    + " " + method[i].getReturnType().getSimpleName() + " " + method[i].getName() + "(");  
  51.                 for (int j = 0; j < parameter.length; j++) {  
  52.                     System.out.print(parameter[j].getSimpleName() + " args");  
  53.                     if (j != parameter.length - 1) {  
  54.                         System.out.print(",");  
  55.                     }  
  56.                 }  
  57.                 System.out.print(") ");  
  58.                 Class exception[] = method[i].getExceptionTypes();  
  59.  
  60.                 if (exception.length > 0) {  
  61.                     System.out.print("throws ");  
  62.                     for (int j = 0; j < exception.length; j++) {  
  63.                         System.out.print(exception[j].getSimpleName());  
  64.                     }  
  65.                 }  
  66.                 System.out.println("{");  
  67.                 System.out.println("\t... ...");  
  68.                 System.out.println("}");  
  69.             }  
  70.         }  
  71.     }  

 

1. 得到某个对象的属性

1 public Object getProperty(Object owner, String fieldName) throws Exception {
2     Class ownerClass = owner.getClass();

4     Field field = ownerClass.getField(fieldName);

6     Object property = field.get(owner);

8     return property;
9 }
  Class ownerClass = owner.getClass():得到该对象的Class。

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

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

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

 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. 执行某对象的方法

 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):通过Method名和参数的Class数组得到要执行的Method。

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

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

 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. 新建实例
 1 
 2 public Object newInstance(String className, Object[] args) throws Exception {
 3     Class newoneClass = 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     Constructor cons = newoneClass.getConstructor(argsClass);
12 
13     return cons.newInstance(args);
14 
15 }

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

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

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

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

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

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

1 public boolean isInstance(Object obj, Class cls) {
2     return cls.isInstance(obj);
3 }

7. 得到数组中的某个元素
1 public Object getByArray(Object array, int index) {
2     return Array.get(array,index);
3 }
 

 

 


你可能感兴趣的:(java反射机制)