1. 反射概念
??Java反射機制是在運行狀態(tài)中页藻,對于任意一個類植兰,都能夠知道這個類的所有屬性和方法;對于任意一個對象楣导,都能夠調(diào)用它的任意方法和屬性;這種動態(tài)獲取信息以及動態(tài)調(diào)用對象方法的功能稱為Java語言的反射機制噩凹。
反射的使用主要包括四個方面:
- Class,Class類的實例表示正在運行的Java應用程序中的類與接口
- Constructor驮宴,關于類的單個構(gòu)造方法的信息以及對它的訪問權(quán)限
- Field,F(xiàn)ield提供有關類或接口的單個字段的信息幻赚,以及對它動態(tài)訪問權(quán)限
- Method,Method 提供關于類或接口上單獨某個方法的信息
2. 使用示例
??首先構(gòu)造一個用于示范的Person類:
package classTest;
public class Person {
public String name;
private String sex;
public Person(){
}
public Person(String name,String sex){
this.name = name;
this.sex = sex;
}
public void sayHello(){
System.out.println("hello !");
}
// 私有方法
private void run(){
System.out.println("run !");
}
// 私有帶參方法
private void eat(String foodName){
System.out.println("eat " + foodName + " !");
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
public Person(){
}
public Person(String name,String sex){
this.name = name;
this.sex = sex;
}
public void sayHello(){
System.out.println("hello !");
}
}
2.1 Class
??反射提供了三種方法獲取類的Class。
1佳谦、類名.class:
??直接通過已知類名獲取類的Class
Class clazzPerson1 = Person.class;
2、對象.getClass():
??需要獲得具體的對象啥刻,并使用父類的Object.getClass()方法獲取
Person p = new Person();
Class clazzPerson2 = p.getClass();
3咪笑、Class.forName("包名.類名"):
??forName()用于加載類字節(jié)碼到內(nèi)存中,并封裝成一個Class對象
try {
Class clazzPerson3 = Class.forName("classTest.Person");
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
2.2 Constructor
1窗怒、得到String類的所有構(gòu)造方法:
Constructor[] constructors = Class.forName("java.lang.String").getConstructors();
2、得到String類指定的構(gòu)造方法并調(diào)用:
Constructor constructor = Class.forName("java.lang.String").getConstructor(String.class);
String str = (String)constructor.newInstance("abc");
3努隙、Class類的newInstance() 方法用來調(diào)用類的默認構(gòu)造方法:
String obj = (String)Class.forName("java.lang.String").newInstance();
2.3 Field
??Field類代表某個類中的一個成員變量辜昵,并提供動態(tài)的訪問權(quán)限。
1堪置、得到所有的成員變量:
Field[] fields = clazz.getFields(); // 獲取所有public屬性
Field[] fields = clazz.getDeclaredFields(); // 取得所有聲明的屬性
2舀锨、得到指定的成員變量:
Field name = clazz.getField("name");
Field name = clazz.getDeclaredField("name");
3岭洲、設置Field變量是否可以訪問:
field.setAccessible(boolean);
4雁竞、Field變量值的讀取并設置:
field.get(obj);
field.set(obj,value);
測試用例:
@Test
public void test3() throws Exception {
// 測試公有屬性
Class clazz = Class.forName("classTest.Person");
Person person = (Person) clazz.getConstructor().newInstance();
Field fieldName = clazz.getField("name");
fieldName.set(person,"李四");
System.out.println(fieldName.get(person));
// 測試私有屬性
Person person2 = (Person) clazz.getConstructor().newInstance();
Field fieldSex = clazz.getDeclaredField("sex");
fieldSex.setAccessible(true);
fieldSex.set(person2,"女");
System.out.println(fieldSex.get(person2));
}
2.4 Method
??Method 類代表某個類中的一個成員方法碑诉。
1、獲得所有方法:
getDeclaredMethods()
getMethods()
2进栽、獲得指定的方法:
//name為方法名稱
getDeclredMethods(String name,Class<?>...parameterTypes)
getMethods(String name,Class<?>...parameterTypes)
3、通過反射執(zhí)行方法:
invoke(Object obj,Object...args)
Method類可以對象可以獲取并操作屬于一個類的所有方法:
@Test
public void test4() throws Exception {
// 測試共有方法
Class clazz = Class.forName("classTest.Person");
Person person = (Person) clazz.getConstructor().newInstance();
Method method = clazz.getMethod("sayHello");
method.invoke(person);
// 測試私有方法
Method method2 = clazz.getDeclaredMethod("run");
method2.setAccessible(true);
method2.invoke(person);
// 測試私有帶參的方法
Method method3 = clazz.getDeclaredMethod("eat", String.class);
method3.setAccessible(true);
method3.invoke(person,"apple");
}