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 |