這期內(nèi)容當(dāng)中小編將會給大家?guī)碛嘘P(guān)java中的反射機(jī)制介紹,以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。
創(chuàng)新互聯(lián)建站是一家集網(wǎng)站建設(shè),周寧企業(yè)網(wǎng)站建設(shè),周寧品牌網(wǎng)站建設(shè),網(wǎng)站定制,周寧網(wǎng)站建設(shè)報價,網(wǎng)絡(luò)營銷,網(wǎng)絡(luò)優(yōu)化,周寧網(wǎng)站推廣為一體的創(chuàng)新建站企業(yè),幫助傳統(tǒng)企業(yè)提升企業(yè)形象加強(qiáng)企業(yè)競爭力。可充分滿足這一群體相比中小企業(yè)更為豐富、高端、多元的互聯(lián)網(wǎng)需求。同時我們時刻保持專業(yè)、時尚、前沿,時刻以成就客戶成長自我,堅持不斷學(xué)習(xí)、思考、沉淀、凈化自己,讓我們?yōu)楦嗟钠髽I(yè)打造出實(shí)用型網(wǎng)站。
一、什么是JAVA的反射機(jī)制
主要是指程序可以訪問,檢測和修改它本身狀態(tài)或行為的一種能力,并能根據(jù)自身行為的狀態(tài)和結(jié)果,調(diào)整或修改應(yīng)用所描述行為的狀態(tài)和相關(guān)的語義。
反射是java中一種強(qiáng)大的工具,能夠使我們很方便的創(chuàng)建靈活的代碼,這些代碼可以再運(yùn)行時裝配,無需在組件之間進(jìn)行源代碼鏈接。這個機(jī)制允許程序在運(yùn)行時透過Reflection APIs取得任何一個已知名稱的class的內(nèi)部信息,包括其modifiers(諸如public, static 等)、superclass(例如Object)、實(shí)現(xiàn)之interfaces(例如Cloneable),也包括fields和methods的所有信息,并可于運(yùn)行時改變fields內(nèi)容或喚起methods。
Java反射機(jī)制容許程序在運(yùn)行時加載、探知、使用編譯期間完全未知的classes。
換言之,Java可以加載一個運(yùn)行時才得知名稱的class,獲得其完整結(jié)構(gòu)。
二、JDK中提供的Reflection API
Java反射相關(guān)的API在包java.lang.reflect中,JDK 1.6.0的reflect包如下圖:
Member接口 | 該接口可以獲取有關(guān)類成員(域或者方法)后者構(gòu)造函數(shù)的信息。 |
AccessibleObject類 | 該類是域(field)對象、方法(method)對象、構(gòu)造函數(shù)(constructor)對象的基礎(chǔ)類。它提供了將反射的對象標(biāo)記為在使用時取消默認(rèn) Java 語言訪問控制檢查的能力。 |
Array類 | 該類提供動態(tài)地生成和訪問JAVA數(shù)組的方法。 |
Constructor類 | 提供一個類的構(gòu)造函數(shù)的信息以及訪問類的構(gòu)造函數(shù)的接口。 |
Field類 | 提供一個類的域的信息以及訪問類的域的接口。 |
Method類 | 提供一個類的方法的信息以及訪問類的方法的接口。 |
Modifier類 | 提供了 static 方法和常量,對類和成員訪問修飾符進(jìn)行解碼。 |
Proxy類 | 提供動態(tài)地生成代理類和類實(shí)例的靜態(tài)方法。 |
三、JAVA反射機(jī)制提供了什么功能
Java反射機(jī)制提供如下功能:
在運(yùn)行時判斷任意一個對象所屬的類
在運(yùn)行時構(gòu)造任意一個類的對象
在運(yùn)行時判段任意一個類所具有的成員變量和方法
在運(yùn)行時調(diào)用任一個對象的方法
在運(yùn)行時創(chuàng)建新類對象
在使用Java的反射功能時,基本首先都要獲取類的Class對象,再通過Class對象獲取其他的對象。
這里首先定義用于測試的類:
class Type{ public int pubIntField; public String pubStringField; private int prvIntField; public Type(){ Log("Default Constructor"); } Type(int arg1, String arg2){ pubIntField = arg1; pubStringField = arg2; Log("Constructor with parameters"); } public void setIntField(int val) { this.prvIntField = val; } public int getIntField() { return prvIntField; } private void Log(String msg){ System.out.println("Type:" + msg); } } class ExtendType extends Type{ public int pubIntExtendField; public String pubStringExtendField; private int prvIntExtendField; public ExtendType(){ Log("Default Constructor"); } ExtendType(int arg1, String arg2){ pubIntExtendField = arg1; pubStringExtendField = arg2; Log("Constructor with parameters"); } public void setIntExtendField(int field7) { this.prvIntExtendField = field7; } public int getIntExtendField() { return prvIntExtendField; } private void Log(String msg){ System.out.println("ExtendType:" + msg); } }
1、獲取類的Class對象
Class 類的實(shí)例表示正在運(yùn)行的 Java 應(yīng)用程序中的類和接口。獲取類的Class對象有多種方式:
調(diào)用getClass:
Boolean var1 = true; Class> classType2 = var1.getClass(); System.out.println(classType2);
輸出:class java.lang.Boolean
運(yùn)用.class 語法:
Class> classType4 = Boolean.class; System.out.println(classType4);
輸出:class java.lang.Boolean
運(yùn)用static method Class.forName():
Class> classType5 = Class.forName("java.lang.Boolean"); System.out.println(classType5);
輸出:class java.lang.Boolean
運(yùn)用primitive wrapper classes的TYPE 語法:
這里返回的是原生類型,和Boolean.class返回的不同
Class> classType3 = Boolean.TYPE; System.out.println(classType3);
輸出:boolean
2、獲取類的Fields
可以通過反射機(jī)制得到某個類的某個屬性,然后改變對應(yīng)于這個類的某個實(shí)例的該屬性值。JAVA 的Class
public Field getField(String name) | 返回一個 Field 對象,它反映此 Class 對象所表示的類或接口的指定公共成員字段 |
public Field[] getFields() | 返回一個包含某些 Field 對象的數(shù)組,這些對象反映此 Class 對象所表示的類或接口的所有可訪問公共字段 |
public Field getDeclaredField(String name) | 返回一個 Field 對象,該對象反映此 Class 對象所表示的類或接口的指定已聲明字段 |
public Field[] getDeclaredFields() | 返回 Field 對象的一個數(shù)組,這些對象反映此 Class 對象所表示的類或接口所聲明的所有字段 |
Class> classType = ExtendType.class; // 使用getFields獲取屬性 Field[] fields = classType.getFields(); for (Field f : fields) { System.out.println(f); } System.out.println(); // 使用getDeclaredFields獲取屬性 fields = classType.getDeclaredFields(); for (Field f : fields) { System.out.println(f); }
輸出:
public int com.quincy.ExtendType.pubIntExtendField
public java.lang.String com.quincy.ExtendType.pubStringExtendField
public int com.quincy.Type.pubIntField
public java.lang.String com.quincy.Type.pubStringField
public int com.quincy.ExtendType.pubIntExtendField
public java.lang.String com.quincy.ExtendType.pubStringExtendField
private int com.quincy.ExtendType.prvIntExtendField
可見getFields和getDeclaredFields區(qū)別:
getFields返回的是申明為public的屬性,包括父類中定義,
getDeclaredFields返回的是指定類定義的所有定義的屬性,不包括父類的。
3、獲取類的Method
通過反射機(jī)制得到某個類的某個方法,然后調(diào)用對應(yīng)于這個類的某個實(shí)例的該方法
Class
public Method getMethod(String name, Class>... parameterTypes)
返回一個 Method 對象,它反映此 Class 對象所表示的類或接口的指定公共成員方法
public Method[] getMethods()
返回一個包含某些 Method 對象的數(shù)組,這些對象反映此 Class 對象所表示的類或接口(包括那些由該類或接口聲明的以及從超類和超接口繼承的那些的類或接口)的公共 member 方法
public Method getDeclaredMethod(String name,Class>... parameterTypes)
返回一個 Method 對象,該對象反映此 Class 對象所表示的類或接口的指定已聲明方法
public Method[] getDeclaredMethods()
返回 Method 對象的一個數(shù)組,這些對象反映此 Class 對象表示的類或接口聲明的所有方法,包括公共、保護(hù)、默認(rèn)(包)訪問和私有方法,但不包括繼承的方法
// 使用getMethods獲取函數(shù) Class> classType = ExtendType.class; Method[] methods = classType.getMethods(); for (Method m : methods) { System.out.println(m); } System.out.println(); // 使用getDeclaredMethods獲取函數(shù) methods = classType.getDeclaredMethods(); for (Method m : methods) { System.out.println(m); }
輸出:
public void com.quincy.ExtendType.setIntExtendField(int)
public int com.quincy.ExtendType.getIntExtendField()
public void com.quincy.Type.setIntField(int)
public int com.quincy.Type.getIntField()
public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException
public final void java.lang.Object.wait() throws java.lang.InterruptedException
public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException
public boolean java.lang.Object.equals(java.lang.Object)
public java.lang.String java.lang.Object.toString()
public native int java.lang.Object.hashCode()
public final native java.lang.Class java.lang.Object.getClass()
public final native void java.lang.Object.notify()
public final native void java.lang.Object.notifyAll()
private void com.quincy.ExtendType.Log(java.lang.String)
public void com.quincy.ExtendType.setIntExtendField(int)
public int com.quincy.ExtendType.getIntExtendField()
4、獲取類的Constructor
通過反射機(jī)制得到某個類的構(gòu)造器,然后調(diào)用該構(gòu)造器創(chuàng)建該類的一個實(shí)例
Class
public ConstructorgetConstructor(Class>... parameterTypes)
返回一個 Constructor 對象,它反映此 Class 對象所表示的類的指定公共構(gòu)造方法
public Constructor>[] getConstructors()
返回一個包含某些 Constructor 對象的數(shù)組,這些對象反映此 Class 對象所表示的類的所有公共構(gòu)造方法
public ConstructorgetDeclaredConstructor(Class>... parameterTypes)
返回一個 Constructor 對象,該對象反映此 Class 對象所表示的類或接口的指定構(gòu)造方法
public Constructor>[] getDeclaredConstructors()
返回 Constructor 對象的一個數(shù)組,這些對象反映此 Class 對象表示的類聲明的所有構(gòu)造方法。它們是公共、保護(hù)、默認(rèn)(包)訪問和私有構(gòu)造方法
// 使用getConstructors獲取構(gòu)造器 Constructor>[] constructors = classType.getConstructors(); for (Constructor> m : constructors) { System.out.println(m); } System.out.println(); // 使用getDeclaredConstructors獲取構(gòu)造器 constructors = classType.getDeclaredConstructors(); for (Constructor> m : constructors) { System.out.println(m); } 輸出: public com.quincy.ExtendType() public com.quincy.ExtendType() com.quincy.ExtendType(int,java.lang.String)
5、新建類的實(shí)例
通過反射機(jī)制創(chuàng)建新類的實(shí)例,有幾種方法可以創(chuàng)建
調(diào)用無自變量ctor:
1、調(diào)用類的Class對象的newInstance方法,該方法會調(diào)用對象的默認(rèn)構(gòu)造器,如果沒有默認(rèn)構(gòu)造器,會調(diào)用失敗.
Class> classType = ExtendType.class; Object inst = classType.newInstance(); System.out.println(inst);
輸出:
Type:Default Constructor
ExtendType:Default Constructor
com.quincy.ExtendType@d80be3
2、調(diào)用默認(rèn)Constructor對象的newInstance方法
Class> classType = ExtendType.class; Constructor> constructor1 = classType.getConstructor(); Object inst = constructor1.newInstance(); System.out.println(inst);
輸出:
Type:Default Constructor
ExtendType:Default Constructor
com.quincy.ExtendType@1006d75
調(diào)用帶參數(shù)ctor:
3、調(diào)用帶參數(shù)Constructor對象的newInstance方法
Constructor> constructor2 = classType.getDeclaredConstructor(int.class, String.class); Object inst = constructor2.newInstance(1, "123"); System.out.println(inst);
輸出:
Type:Default Constructor
ExtendType:Constructor with parameters
com.quincy.ExtendType@15e83f9
6、調(diào)用類的函數(shù)
通過反射獲取類Method對象,調(diào)用Field的Invoke方法調(diào)用函數(shù)。
Class> classType = ExtendType.class; Object inst = classType.newInstance(); Method logMethod = classType.getDeclaredMethod("Log", String.class); logMethod.invoke(inst, "test"); 輸出: Type:Default Constructor ExtendType:Default Constructor Class com.quincy.ClassT can not access a member of class com.quincy.ExtendType with modifiers "private" 上面失敗是由于沒有權(quán)限調(diào)用private函數(shù),這里需要設(shè)置Accessible為true; Class> classType = ExtendType.class; Object inst = classType.newInstance(); Method logMethod = classType.getDeclaredMethod("Log", String.class); logMethod.setAccessible(true); logMethod.invoke(inst, "test");
7、設(shè)置/獲取類的屬性值
通過反射獲取類的Field對象,調(diào)用Field方法設(shè)置或獲取值
Class> classType = ExtendType.class; Object inst = classType.newInstance(); Field intField = classType.getField("pubIntExtendField"); intField.setInt(inst, 100); int value = intField.getInt(inst);
四、動態(tài)創(chuàng)建代理類
代理模式:代理模式的作用=為其他對象提供一種代理以控制對這個對象的訪問。
代理模式的角色:
抽象角色:聲明真實(shí)對象和代理對象的共同接口
代理角色:代理角色內(nèi)部包含有真實(shí)對象的引用,從而可以操作真實(shí)對象。
真實(shí)角色:代理角色所代表的真實(shí)對象,是我們最終要引用的對象。
動態(tài)代理:
java.lang.reflect.Proxy:
Proxy 提供用于創(chuàng)建動態(tài)代理類和實(shí)例的靜態(tài)方法,它還是由這些方法創(chuàng)建的所有動態(tài)代理類的超類
InvocationHandler:
是代理實(shí)例的調(diào)用處理程序 實(shí)現(xiàn)的接口,每個代理實(shí)例都具有一個關(guān)聯(lián)的調(diào)用處理程序。對代理實(shí)例調(diào)用方法時,將對方法調(diào)用進(jìn)行編碼并將其指派到它的調(diào)用處理程序的 invoke 方法。
動態(tài)Proxy是這樣的一種類:
它是在運(yùn)行生成的類,在生成時你必須提供一組Interface給它,然后該class就宣稱它實(shí)現(xiàn)了這些interface。你可以把該class的實(shí)例當(dāng)作這些interface中的任何一個來用。當(dāng)然,這個Dynamic Proxy其實(shí)就是一個Proxy,它不會替你作實(shí)質(zhì)性的工作,在生成它的實(shí)例時你必須提供一個handler,由它接管實(shí)際的工作。
在使用動態(tài)代理類時,我們必須實(shí)現(xiàn)InvocationHandler接口
步驟:
1、定義抽象角色
public interface Subject { public void Request(); }
2、定義真實(shí)角色
public class RealSubject implements Subject { @Override public void Request() { // TODO Auto-generated method stub System.out.println("RealSubject"); } }
3、定義代理角色
public class DynamicSubject implements InvocationHandler { private Object sub; public DynamicSubject(Object obj){ this.sub = obj; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { // TODO Auto-generated method stub System.out.println("Method:"+ method + ",Args:" + args); method.invoke(sub, args); return null; } }
4、通過Proxy.newProxyInstance構(gòu)建代理對象
RealSubject realSub = new RealSubject(); InvocationHandler handler = new DynamicSubject(realSub); Class> classType = handler.getClass(); Subject sub = (Subject)Proxy.newProxyInstance(classType.getClassLoader(), realSub.getClass().getInterfaces(), handler); System.out.println(sub.getClass());
5、通過調(diào)用代理對象的方法去調(diào)用真實(shí)角色的方法。
sub.Request();
輸出:
class $Proxy0 新建的代理對象,它實(shí)現(xiàn)指定的接口
Method:public abstract void DynamicProxy.Subject.Request(),Args:null
RealSubject 調(diào)用的真實(shí)對象的方法
上述就是小編為大家分享的java中的反射機(jī)制了,如果您也有類似的疑惑,不妨參照上述方法進(jìn)行嘗試。如果想了解更多相關(guān)內(nèi)容,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊。