『面試高頻』Java:反射機(jī)制之通過Class獲取一個類身上的所有方法

問題

在運(yùn)行時,對一個JAVA類鹰贵,能否知道屬性和方法;能否調(diào)用它的任意方法康嘉?

答案是可以的碉输,JAVA提供一種反射機(jī)制可以實(shí)現(xiàn)。


本人另外整理了我從零開始自學(xué)Java學(xué)習(xí)路線亭珍、學(xué)習(xí)資料及近五年大廠的面試真題大全敷钾,如果需要的話自行領(lǐng)取下載:騰訊文檔

一、什么是JAVA的反射機(jī)制

Java反射是Java被視為動態(tài)(或準(zhǔn)動態(tài))語言的一個關(guān)鍵性質(zhì)肄梨。這個機(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對象有多種方式:

2刁憋、獲取類的Fields

可以通過反射機(jī)制得到某個類的某個屬性滥嘴,然后改變對應(yīng)于這個類的某個實(shí)例的該屬性值。JAVA 的Class<T>類提供了幾個方法獲取類的屬性至耻。

public FieldgetField(String name) 返回一個 Field 對象若皱,它反映此 Class 對象所表示的類或接口的指定公共成員字段
public Field[] getFields() 返回一個包含某些 Field 對象的數(shù)組,這些對象反映此 Class 對象所表示的類或接口的所有可訪問公共字段
public FieldgetDeclaredField(Stringname) 返回一個 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<T>類提供了幾個方法獲取類的方法。

public MethodgetMethod(String name, Class<?>... parameterTypes) 返回一個 Method 對象尤莺,它反映此 Class 對象所表示的類或接口的指定公共成員方法
public Method[] getMethods() 返回一個包含某些 Method 對象的數(shù)組旅敷,這些對象反映此 Class 對象所表示的類或接口(包括那些由該類或接口聲明的以及從超類和超接口繼承的那些的類或接口)的公共 member 方法
public MethodgetDeclaredMethod(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<T>類提供了幾個方法獲取類的構(gòu)造器锤躁。

public Constructor<T> getConstructor(Class<?>... parameterTypes) 返回一個 Constructor 對象,它反映此 Class 對象所表示的類的指定公共構(gòu)造方法
public Constructor<?>[] getConstructors() 返回一個包含某些 Constructor 對象的數(shù)組或详,這些對象反映此 Class 對象所表示的類的所有公共構(gòu)造方法
public Constructor<T> getDeclaredConstructor(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)建

6、調(diào)用類的函數(shù)

通過反射獲取類Method對象,調(diào)用Field的Invoke方法調(diào)用函數(shù)嗡善。

Class<?> classType = ExtendType.class;
Object inst = classType.newInstance();
Method logMethod = classType.<strong>getDeclaredMethod</strong>("Log", String.class);
logMethod.invoke(inst, "test");
 
輸出:
Type:Default Constructor
ExtendType:Default Constructor
<font color="#ff0000">Class com.quincy.ClassT can not access a member of class com.quincy.ExtendType with modifiers "private"</font>
 
<font color="#ff0000">上面失敗是由于沒有權(quán)限調(diào)用private函數(shù)辑莫,這里需要設(shè)置Accessible為true;</font>
Class<?> classType = ExtendType.class;
Object inst = classType.newInstance();
Method logMethod = classType.getDeclaredMethod("Log", String.class);
<font color="#ff0000">logMethod.setAccessible(true);</font>
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.<strong>setInt</strong>(inst, 100);
int value = intField.<strong>getInt</strong>(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知識分享就到這里嘲更!點(diǎn)關(guān)注,不迷路揩瞪,關(guān)注程序員曾曾赋朦,每天分享不同的Java基礎(chǔ)知識

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市宠哄,隨后出現(xiàn)的幾起案子壹将,更是在濱河造成了極大的恐慌,老刑警劉巖毛嫉,帶你破解...
    沈念sama閱讀 211,194評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件诽俯,死亡現(xiàn)場離奇詭異,居然都是意外死亡承粤,警方通過查閱死者的電腦和手機(jī)暴区,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,058評論 2 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來密任,“玉大人颜启,你說我怎么就攤上這事±嘶洌” “怎么了缰盏?”我有些...
    開封第一講書人閱讀 156,780評論 0 346
  • 文/不壞的土叔 我叫張陵,是天一觀的道長淹遵。 經(jīng)常有香客問我口猜,道長,這世上最難降的妖魔是什么透揣? 我笑而不...
    開封第一講書人閱讀 56,388評論 1 283
  • 正文 為了忘掉前任济炎,我火速辦了婚禮,結(jié)果婚禮上辐真,老公的妹妹穿的比我還像新娘须尚。我一直安慰自己,他們只是感情好侍咱,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,430評論 5 384
  • 文/花漫 我一把揭開白布耐床。 她就那樣靜靜地躺著,像睡著了一般楔脯。 火紅的嫁衣襯著肌膚如雪撩轰。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,764評論 1 290
  • 那天昧廷,我揣著相機(jī)與錄音堪嫂,去河邊找鬼。 笑死木柬,一個胖子當(dāng)著我的面吹牛皆串,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播眉枕,決...
    沈念sama閱讀 38,907評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼恶复,長吁一口氣:“原來是場噩夢啊……” “哼娇唯!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起寂玲,我...
    開封第一講書人閱讀 37,679評論 0 266
  • 序言:老撾萬榮一對情侶失蹤塔插,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后拓哟,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體想许,經(jīng)...
    沈念sama閱讀 44,122評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,459評論 2 325
  • 正文 我和宋清朗相戀三年断序,在試婚紗的時候發(fā)現(xiàn)自己被綠了流纹。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,605評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡违诗,死狀恐怖漱凝,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情诸迟,我是刑警寧澤茸炒,帶...
    沈念sama閱讀 34,270評論 4 329
  • 正文 年R本政府宣布,位于F島的核電站阵苇,受9級特大地震影響壁公,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜绅项,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,867評論 3 312
  • 文/蒙蒙 一紊册、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧快耿,春花似錦囊陡、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,734評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至铺浇,卻和暖如春痢畜,著一層夾襖步出監(jiān)牢的瞬間垛膝,已是汗流浹背鳍侣。 一陣腳步聲響...
    開封第一講書人閱讀 31,961評論 1 265
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留吼拥,地道東北人倚聚。 一個月前我還...
    沈念sama閱讀 46,297評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像凿可,于是被迫代替她去往敵國和親惑折。 傳聞我的和親對象是個殘疾皇子授账,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,472評論 2 348

推薦閱讀更多精彩內(nèi)容