通俗的說,反射就是可以獲得類的信息,比如類里面有什么方法、屬性、構(gòu)造函數(shù)等,也可以對類實(shí)例化(不是所有的實(shí)例化都是用new,new必須知道這個類是什么,而實(shí)際情況中很多時候是不能預(yù)先知道這個類名),很多框架都是使用反射的原理,比如spring
1. 反射的概念
主要是指程序可以訪問,檢測和修改它本身狀態(tài)或行為的一種能力,并能根據(jù)自身行為的狀態(tài)和結(jié)果,調(diào)整或修改應(yīng)用所描述行為的狀態(tài)和相關(guān)的語義。
反射是java中一種強(qiáng)大的工具,能夠使我們很方便的創(chuàng)建靈活的代碼,這些代碼可以再運(yùn)行時裝配,無需在組件之間進(jìn)行源代碼鏈接。但是反射使用不當(dāng)會成本很高?。?/p>
2. 反射機(jī)制的作用:
- 反編譯:.class-->.java
- 通過反射機(jī)制訪問java對象的屬性,方法,構(gòu)造方法等;
3. java我們提供了那些反射機(jī)制中的類:
java.lang.Class;
java.lang.reflect.Constructor;
java.lang.reflect.Field;
java.lang.reflect.Method;
java.lang.reflect.Modifier;
強(qiáng)調(diào)的是API是我們最好的老師。
4. 具體功能實(shí)現(xiàn):
- 反射機(jī)制獲取類有三種方法,我們來獲取Employee類型
//第一種方式:
Classc1 = Class.forName("Employee");
//第二種方式:
//java中每個類型都有class 屬性.
Classc2 = Employee.class;
//第三種方式:
//java語言中任何一個java對象都有g(shù)etClass 方法
Employeee = new Employee();
Classc3 = e.getClass(); //c3是運(yùn)行時類 (e的運(yùn)行時類是Employee)
- 創(chuàng)建對象:獲取類以后我們來創(chuàng)建它的對象,利用newInstance:
Class c =Class.forName("Employee");
//創(chuàng)建此Class 對象所表示的類的一個新實(shí)例
Objecto = c.newInstance(); //調(diào)用了Employee的無參數(shù)構(gòu)造方法.
- 獲取屬性:分為所有的屬性和指定的屬性:
a. 先看獲取所有的屬性的寫法:
//獲取整個類
Class c = Class.forName("java.lang.Integer");
//獲取所有的屬性?
Field[] fs = c.getDeclaredFields();
//定義可變長的字符串,用來存儲屬性
StringBuffer sb = new StringBuffer();
//通過追加的方法,將每個屬性拼接到此字符串中
//最外邊的public定義
sb.append(Modifier.toString(c.getModifiers()) + " class " + c.getSimpleName() +"{\n");
//里邊的每一個屬性
for(Field field:fs){
sb.append("\t");//空格
sb.append(Modifier.toString(field.getModifiers())+" ");//獲得屬性的修飾符,例如public,static等等
sb.append(field.getType().getSimpleName() + " ");//屬性的類型的名字
sb.append(field.getName()+";\n");//屬性的名字+回車
}
sb.append("}");
System.out.println(sb);
b.獲取特定的屬性,對比著傳統(tǒng)的方法來學(xué)習(xí):
public static void main(String[] args) throws Exception{
<span style="white-space:pre"> </span>//以前的方式:
/*
User u = new User();
u.age = 12; //set
System.out.println(u.age); //get
*/
//獲取類
Class c = Class.forName("User");
//獲取id屬性
Field idF = c.getDeclaredField("id");
//實(shí)例化這個類賦給o
Object o = c.newInstance();
//打破封裝
idF.setAccessible(true); //使用反射機(jī)制可以打破封裝性,導(dǎo)致了java對象的屬性不安全。
//給o對象的id屬性賦值"110"
idF.set(o, "110"); //set
//get
System.out.println(idF.get(o));
}
方法關(guān)鍵字 | 含義 |
---|---|
getDeclaredMethods() | 獲取所有的方法 |
getReturnType() | 獲得方法的放回類型 |
getParameterTypes() | 獲得方法的傳入?yún)?shù)類型 |
getDeclaredMethod("方法名",參數(shù)類型.class,……) | 獲得特定的方法 |
4,獲取方法,和構(gòu)造方法,不再詳細(xì)描述,只來看一下關(guān)鍵字:
方法關(guān)鍵字 | 含義 |
---|---|
getDeclaredMethods() | 獲取所有的方法 |
getReturnType() | 獲得方法的放回類型 |
getParameterTypes() | 獲得方法的傳入?yún)?shù)類型 |
getDeclaredMethod("方法名",參數(shù)類型.class,……) | 獲得特定的方法 |
構(gòu)造方法關(guān)鍵字 | 含義 |
---|---|
getDeclaredConstructors() | 獲取所有的構(gòu)造方法 |
getDeclaredConstructor(參數(shù)類型.class,……) | 獲取特定的構(gòu)造方法 |
父類和父接口 | 含義 |
---|---|
getSuperclass() | 獲取某類的父類 |
getInterfaces() | 獲取某類實(shí)現(xiàn)的接口 |
這樣我們就可以獲得類的各種內(nèi)容,進(jìn)行了反編譯。對于JAVA這種先編譯再運(yùn)行的語言來說,反射機(jī)制可以使代碼更加靈活,更加容易實(shí)現(xiàn)面向?qū)ο蟆?/p>