Java中的反射(Class類的使用符喝,獲取類的信息闪彼,方法使用等)

Class類的使用

  1. Java中,靜態(tài)成員协饲、普通數(shù)據(jù)類型不是對象备蚓,靜態(tài)成員是屬于類的,而不是屬于某個(gè)對象的

  2. 類是對象囱稽,類是java.lang.Class類的實(shí)例對象

    public class ClassDemo{
    public static void main(String[] args){
    //Foo的實(shí)例對象
    Foo foo1 = new Foo();
    //Foo這個(gè)類 本身也是一個(gè)實(shí)例對象 它是Class類的實(shí)例對象
    //任何一個(gè)類都是Class類的實(shí)例對象,這個(gè)實(shí)例對象有三種表示方式
    //第一種-->實(shí)際告訴我們?nèi)魏我粋€(gè)類都有一個(gè)隱含的靜態(tài)成員變量class
    Class c1 = Foo.class;

       //第二種-->通過該類的對象的getClass()方法得到
       Class c2 = foo1.getClass();
       
       //c1,c2表示了Foo類的類類型(class type)
       //一個(gè)類只可能是Class類的一個(gè)實(shí)例對象
       
       //第三種
       Class c3 = null;
       try{
           c3 = Class.forName("xxx.Foo");
       }catch(ClassNotFoundException e){
           e.printStackTrace();
       }
       //c1 = c2 = c3
       
       //通過該類的類類型創(chuàng)建該類的對象實(shí)例
       Foo foo = (Foo)c1.newInstance();//需要有無參數(shù)的構(gòu)造方法
     }
    

    }
    class Foo{}

  3. 所有的數(shù)據(jù)類型(int, String, Integer)以及void都有自己的類類型二跋,通過int.class可以得到相應(yīng)的類類型

Java動態(tài)加載類

編譯時(shí)加載類時(shí)靜態(tài)加載類战惊,運(yùn)行時(shí)加載類時(shí)動態(tài)加載類,可以通過Class.forName()動態(tài)加載類

下面是一個(gè)靜態(tài)加載類的例子:

Class Test{
  public static void main(String[] args){
    if("A".equals(args[0])){
        A a = new A();
        a.start();
    }
    if("B".equals(args[0])){
        B b = new B();
        b.start();
    }
}
}

Class A{
    public static void start(){
        System.out.println("A start");
    }
}

結(jié)果是編譯出錯(cuò)扎即。

new 創(chuàng)建對象是靜態(tài)加載類吞获,在編譯時(shí)刻就加載所有可能用到的類,就算我們寫了A類谚鄙,但是因?yàn)闆]有B類各拷,編譯時(shí)還是不會通過。如果我們想使用更多的類闷营,我們就要在一開始把所有類寫好烤黍,當(dāng)以后想要擴(kuò)展時(shí),就要重新編譯傻盟!

下面通過Class.forName()動態(tài)加載類

class Test1{
    public static void main(String[] args){
        Class c1 = null;
        try{
            c1 = Class.forName(args[0]);
        }catch(ClassNotFoundException e){
          e.printStackTrace();
      } 
    }
}

這樣 編譯就不會報(bào)錯(cuò)速蕊,在我們使用的時(shí)候,傳入想要運(yùn)行的類名稱即可娘赴。

接下來我們可以通過上面提到的newInstance()方法來創(chuàng)建對象规哲,并通過強(qiáng)制類型轉(zhuǎn)換編程我們想要用的對象。但是這里有一個(gè)問題诽表,就是我們不知道我們傳入的究竟是A類還是B類唉锌,所以在做強(qiáng)制類型轉(zhuǎn)換的時(shí)候并不知道要轉(zhuǎn)換成什么,解決辦法就是為他們創(chuàng)建一個(gè)接口竿奏,實(shí)現(xiàn)其共有的方法

MyChar myChar = (MyChar)c1.newInstance();
myChar.start();

其中

interface MyChar(){
    public void start();
}

這時(shí)我們只需要用A類實(shí)現(xiàn)該接口即可

class A implements MyChar{
    public void start(){
        System.out.println("A start");
    }
}

編譯通過袄简,我們不需要用B類即可,當(dāng)我們想用B類的時(shí)候议双,只需要用B類實(shí)現(xiàn)該接口即可痘番。如果我們想使用更多的類,只需要實(shí)現(xiàn)該接口即可,不需要每次都重新編譯汞舱。因此伍纫,在設(shè)計(jì)程序時(shí),可以盡可能的考慮動態(tài)加載類

獲取類的信息

1.獲取方法信息

//首先要得到該類的類類型
Class c = foo1.getClass();
//獲取類的名稱
String name = c.getName();
//獲取類的方法
Methods[] ms = c.getMethods();//萬事萬物皆對象昂芜,每一個(gè)方法都是Method的對象
//getMethods()得到的是所有public函數(shù)莹规,包括從父類繼承來的
//getDeclaredMethods()得到的是該類自己聲明的方法,不論訪問權(quán)限

//獲取方法的信息
for(int i = 0; i<ms.length;i++){
    //得到方法的返回值的類類型泌神,如int.class, String.class
    Class returnType = ms[i].getReturnType();//returnType.getName()即得到返回值的名稱
    //得到方法的名稱
    String methodName = ms[i].getName();
    //獲取參數(shù)類型:得到的是參數(shù)列表的類類型
    Class[] paramTypes = ms[i].getParameterTypes();//遍歷通過getName()即可得到參數(shù)類型名稱
}

下面是獲取某個(gè)具體方法

Method method = XXX.getClass().getMethod(methodName,new Class[0]);
//getMethod第一個(gè)參數(shù)是方法名良漱,第二個(gè)參數(shù)是該方法的參數(shù)類型,按傳入順序,
//因?yàn)榇嬖谕椒煌瑓?shù)這種情況欢际,所以只有同時(shí)指定方法名和參數(shù)類型才能唯一確定一個(gè)方法
//沒有參數(shù)傳入null母市,獲取不傳
//如一個(gè)函數(shù) int test(int a, String b);
//getMethod("Test",int.class,String.class);

得到的是一個(gè)方法對象,通過調(diào)用invoke()函數(shù)來調(diào)用此方法

//函數(shù)原型:Object Java.lang.reflect.Method.invoke(Object receiver, Object... args)

//Method類的invoke(Object obj,Object args[])方法接收的參數(shù)必須為對象损趋,
 //如果參數(shù)為基本類型數(shù)據(jù)患久,必須轉(zhuǎn)換為相應(yīng)的包裝類型的對象。invoke()方法的返回值總是對象浑槽,
  //如果實(shí)際被調(diào)用的方法的返回類型是基本類型數(shù)據(jù)蒋失,那么invoke()方法會把它轉(zhuǎn)換為相應(yīng)的包裝類型的對象,再將其返回

//receiver:該方法所在類的一個(gè)對象
//args: 傳入的參數(shù) 如 100桐玻,“hello”篙挽,沒有就不傳
//下面是一個(gè)完整例子
class Foo{
    public void run(String str) {
        System.out.println("Foo run " + str);
    }
}
public class Demo {

    public static void main(String[] args) {
        // TODO Auto-generated method stub
        Foo foo1 = new Foo();
        Class<? extends Foo> c1 = foo1.getClass();
        System.out.println(c1.getName());
        try {
            Method method = c1.getMethod("run", String.class);
            method.invoke(foo1, "test reflect");
        } catch (NoSuchMethodException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (SecurityException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}

運(yùn)行結(jié)果:

study_refect.Foo
Foo run test reflect

2.獲取成員變量的信息

成員變量也是對象,是java.lang.reflect.Field的對象

Field[] fs = c.getFields();//獲取的是public成員變量信息
//getDeclaredFields()獲取的是該類所有成員變量
for(Field field:fs){
    //得到成員變量的類類型
    Class fieldType = field.getType();
    String typeName = fieldType.getName();
    //得到成員變量的名稱
    String fieldName = field.getName();
}

3.獲取構(gòu)造函數(shù)的信息

構(gòu)造函數(shù)也是對象镊靴,是java.lang.Constructor的對象,獲取方法這里不再贅述

通過反射理解泛型的本質(zhì)(類型擦除)

Java中的泛型是通過類型擦除來實(shí)現(xiàn)的铣卡。所謂類型擦除,是指通過類型參數(shù)合并邑闲,將泛型類型實(shí)例關(guān)聯(lián)到同一份字節(jié)碼上算行。編譯器只為泛型類型生成一份字節(jié)碼,并將其實(shí)例關(guān)聯(lián)到這份字節(jié)碼上苫耸。類型擦除的關(guān)鍵在于從泛型類型中清除類型參數(shù)的相關(guān)信息州邢,并且再必要的時(shí)候添加類型檢查和類型轉(zhuǎn)換的方法。

下面通過兩個(gè)例子來證明在編譯時(shí)確實(shí)發(fā)生了類型擦除褪子。

例1分別創(chuàng)建實(shí)際類型為String和Integer的ArrayList對象量淌,通過getClass()方法獲取兩個(gè)實(shí)例的類,最后判斷這個(gè)實(shí)例的類是相等的嫌褪,證明兩個(gè)實(shí)例共享同一個(gè)類呀枢。

// 聲明一個(gè)具體類型為String的ArrayList
ArrayList<String> arrayList1 = new ArrayList<String>();  
arrayList1.add("abc");  

// 聲明一個(gè)具體類型為Integer的ArrayList
ArrayList<Integer> arrayList2 = new ArrayList<Integer>();  
arrayList2.add(123);  

System.out.println(arrayList1.getClass() == arrayList2.getClass());  // 結(jié)果為true

例2創(chuàng)建一個(gè)只能存儲Integer的ArrayList對象,在add一個(gè)整型數(shù)值后笼痛,利用反射調(diào)用add(Object o)add一個(gè)asd字符串裙秋,此時(shí)運(yùn)行代碼不會報(bào)錯(cuò)琅拌,運(yùn)行結(jié)果會打印出1和asd兩個(gè)值。這時(shí)再里利用反射調(diào)用add(Integer o)方法摘刑,運(yùn)行會拋出codeNoSuchMethodException異常进宝。這充分證明了在編譯后,擦除了Integer這個(gè)泛型信息枷恕,只保留了原始類型党晋。

ArrayList<Integer> arrayList3 = new ArrayList<Integer>();
arrayList3.add(1);
arrayList3.getClass().getMethod("add", Object.class).invoke(arrayList3, "asd");
for (int i = 0; i < arrayList3.size(); i++) {
    System.out.println(arrayList3.get(i)); // 輸出1,asd
}
arrayList3.getClass().getMethod("add", Integer.class).invoke(arrayList3, 2); // NoSuchMethodException:java

Java中集合的泛型徐块,是防止錯(cuò)誤輸入的未玻,只在編譯階段有效,繞過編譯階段就無效了胡控,因此可以通過反射繞過編譯

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末扳剿,一起剝皮案震驚了整個(gè)濱河市,隨后出現(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ī)與錄音休傍,去河邊找鬼征绎。 笑死,一個(gè)胖子當(dāng)著我的面吹牛磨取,可吹牛的內(nèi)容都是我干的人柿。 我是一名探鬼主播,決...
    沈念sama閱讀 38,907評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼忙厌,長吁一口氣:“原來是場噩夢啊……” “哼凫岖!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起逢净,我...
    開封第一講書人閱讀 37,679評論 0 266
  • 序言:老撾萬榮一對情侶失蹤哥放,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后爹土,有當(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
  • 正文 我和宋清朗相戀三年胀茵,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了社露。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,605評論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡琼娘,死狀恐怖峭弟,靈堂內(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. 我叫王不留佛舱,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,297評論 2 360
  • 正文 我出身青樓挨决,卻偏偏與公主長得像请祖,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子脖祈,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,472評論 2 348

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

  • 一肆捕、理解Class對象 (一)RRIT的概念以及Class對象的作用 認(rèn)識Class對象之前,先來了解一個(gè)概念盖高,R...
    守住陽光閱讀 2,046評論 0 0
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理慎陵,服務(wù)發(fā)現(xiàn),斷路器喻奥,智...
    卡卡羅2017閱讀 134,628評論 18 139
  • 在日本深度學(xué)習(xí)了幾天席纽,隱約窺得了日本設(shè)計(jì)的一些“套路”,那就是一定是有融合的撞蚕,兩個(gè)越是截然不同的事物完美融合在一起...
    Ivy1999閱讀 680評論 0 2
  • 雪胆筒,下來了。 盼望著诈豌,盼望著,全國下起了雪抒和。孩子高興地嚷嚷著要堆雪人打雪仗矫渔,老師看著可愛的小臉答應(yīng)了孩子們體育課打...
    Life_53a9閱讀 208評論 1 4