Java 的反射机制是使其具有动态特性的非常关键的一种机制,也是在JavaBean 中广泛应用的一种特性。
简单来说,一个类或者一个对象是拥有下面几种属性的:
Method,Constructor,Field,其大致结构类图如下:
我们现在用代码来说明问题:
首先,我们看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
|