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 |