简单总结Java的反射机制的运用

时间:2022-01-17 01:34:00

Java 的反射机制是使其具有动态特性的非常关键的一种机制,也是在JavaBean 中广泛应用的一种特性。

简单来说,一个类或者一个对象是拥有下面几种属性的:
Method,Constructor,Field,其大致结构类图如下:

简单总结Java的反射机制的运用

我们现在用代码来说明问题:

首先,我们看Class类,在Class类中,我们可以看见下面的几个重要的方法;

  • getInterfaces()
  • getSuperClass();
  • isInterface();

这是用来得到一个类的接口或者超类,以及判断这个类是不是一个接口;

 

  • forName(String className);根据一个类名得到一个相应的类类型;
  • getClassLoader();得到这个类相应的类加载器;

以及下面的几种方法:

?
1
getField(Method/Constructor)(s)(name); getDeclaredField(Method/Constructor)(s)();


分别是获取这个类的相应的Constructor,field,method的;

区别在于,含有Declared的方法能够得到这个类所声明的所有的属性,而没有Declared的只能得到公共public的属性;


而继承了Member分别赋予了这个三个类能够得到声明其的Class,用getDeclaringClass();在这里我们再次介绍一下Modifiers,我们都知道在field或者Constructor,Method前面都含有若干修饰符,如:

?
1
public static final String name="corey";

等等,我们应用getModifiers()能够拿到这个修饰符的一个整形值,然后应用Modifier这个类的静态方法来进行判断;如:
   Modifier.isStatic(int)等等;

接下来,我们来看看AccessibleObject的几个主要的方法,AccessibleObject中主要的几个方式第一是
getAnnotation();得到某个属性的注释;
isAccessible();能否访问;如果不能访问,我们可以采取setAccessible(boolean)来设置其的可访问性;(这个我们在spring中看到过);

然后我们分别来看看这个三个类一些重要的特性:


Constructor:

  • newInstance(args):能够使用这个构造器得到一个类的实例;


Field:

  • getType();得到这个字段的类;
  • set/get(Object,value):一系列的基本类型字段的设置方法或者Object的设置方法;

Method:

  • getParameterTypes();得到所有参数的类型;
  • getExceptionTypes();得到所有抛出异常的类型;
  • invoke(Object,args);调用Object对象的这个方法,参数是args;

下面是一份实例代码:

?
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
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
package org.corey.demo;
public interface IName {
  public String getFirstName();
  public void setFirstName(String firstName);
  public String getLastName();
  public void setLastName(String lastName);
}
 
package org.corey.demo;
public class Name {
  private String firstName;
  private String lastName;
  public String publicName;
  
  public Name(String firstName, String lastName) {
    this.firstName = firstName;
    this.lastName = lastName;
  }
  public Name() {
  }
  public String getFirstName() {
    return firstName;
  }
  public void setFirstName(String firstName) {
    this.firstName = firstName;
  }
  public String getLastName() {
    return lastName;
  }
  public void setLastName(String lastName) {
    this.lastName = lastName;
  }
}
 
 
package org.corey.demo;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
public class Demo {
  /**
   * @param args
   */
  public static void main(String[] args) {
    try {
      Class clazz = Class.forName("org.corey.demo.Name");
      Constructor con = clazz.getDeclaredConstructor(new Class[] {
          String.class, String.class });
      Name corey = (Name)(con.newInstance("corey", "zhou"));
      System.out.println(corey.getFirstName()+" "+corey.getLastName());
      Field[] fields=clazz.getDeclaredFields();
      for(int index=0;index<fields.length;index++){
        System.out.println(fields[index].getName()+" accessible "+fields[index].isAccessible());
        
      }
      Method[] methods=clazz.getDeclaredMethods();
      for(int index=0;index<methods.length;index++){
        System.out.println(methods[index].getName());
      }
      
      Field field=clazz.getDeclaredField("firstName");
      if(!field.isAccessible()){
        field.setAccessible(true);
        field.set(corey, "syna");
      }
      
      Method method=clazz.getDeclaredMethod("setLastName", new Class[]{String.class});
      
      method.invoke(corey, "wang");
      
      System.out.println(corey.getFirstName()+" "+corey.getLastName());
    } catch (Exception e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
  }
}

console:

?
1
2
3
4
5
6
7
8
9
corey zhou
firstName accessible false
lastName accessible false
publicName accessible false
getFirstName
getLastName
setLastName
setFirstName
syna wang