java反射技術

寫在前面

想必開發過接口的童鞋們,應該或多或少寫過一些接口說明文檔。那么,有沒有可能把現有的接口做成一個界面在頁面展現出來而不用去寫什么接口文檔,在頁面展示的信息包括接口名,入參,屬性,注釋…… 如果有提供這些接口信息的池的話就用池的方式來做,我覺得做起來也挺方便的,效率也高。如果項目中沒有這樣的池,可以考慮用反射的方式來實現。那么,今天先來回顧一下java反射機制。關于反射的介紹,了解請看 官方文檔

JAVA反射機制

1、Java反射的概念
反射含義:可以獲取正在運行的Java對象。

詳細解釋見百度百科:

AVA反射機制是在運行狀態中,對于任意一個類,都能夠知道這個類的所有屬性和方法;對于任意一個對象,都能夠調用它的任意一個方法和屬性;這種動態獲取的信息以及動態調用對象的方法的功能稱為java語言的反射機制。 JAVA反射(放射)機制:“程序運行時,允許改變程序結構或變量類型,這種語言稱為動態語言”。從這個觀點看,Perl,Python,Ruby是動態語言,C++,Java,C#不是動態語言。但是JAVA有著一個非常突出的動態相關機制:Reflection,用在Java身上指的是我們可以于運行時加載、探知、使用編譯期間完全未知的classes。換句話說,Java程序可以加載一個運行時才得知名稱的class,獲悉其完整構造(但不包括methods定義),并生成其對象實體、或對其fields設值、或喚起其methods。

2、Java反射的功能
1)可以判斷運行時對象所屬的類
2)可以判斷運行時對象所具有的成員變量和方法
3)通過反射甚至可以調用到private的方法
4)生成動態代理
Java反射的功能,一句話總結就是:反射用于在運行時檢測和修改某個對象的結構及其行為
3、實現Java反射的類
1)Class:它表示正在運行的Java應用程序中的類和接口
2)Field:提供有關類或接口的屬性信息,以及對它的動態訪問權限
3)Constructor:提供關于類的單個構造方法的信息以及對它的訪問權限
4)Method:提供關于類或接口中某個方法信息
注意:Class類是Java反射中最重要的一個功能類,所有獲取對象的信息(包括:方法/屬性/構造方法/訪問權限)都需要它來實現
4、編寫Java反射程序的步驟   1)必須首先獲取一個類的Class對象   例如:   Class c1 = Test.class;   Class c2 = Class.forName(“com.mysql.jdbc.Driver ”);   Class c3 = new Test().getClass();   2)然后分別調用Class對象中的方法來獲取一個類的屬性/方法/構造方法的結構   注意:如果要能夠正常的獲取類中方法/屬性/構造方法應該重點掌握如下的反射類   Field   Constructor   Method
5、例子

<pre>
package wblearn;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;

/**

  • @author wblearn

  • @date 2016-9-25
    */
    public class TestReflection {
    private String name;
    private boolean learn;
    private int[] age;

    public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException,
    SecurityException, NoSuchMethodException, IllegalArgumentException, InvocationTargetException {
    Class c1 = TestReflection.class;
    Class c2 = Class.forName("wblearn.TestReflection");
    //獲取指定的包名
    String package01 = c1.getPackage().getName();
    String package02 = c2.getPackage().getName();
    System.out.println("package01 = " + package01);
    System.out.println("package02 = " + package02);
    //獲取類的修飾符
    int mod = c1.getModifiers();
    String modifier = Modifier.toString(mod);
    System.out.println("modifier = " + modifier);
    //獲取指定類的完全限定名
    String className = c1.getName();
    System.out.println("className = " + className);
    //獲取指定類的父類
    Class superClazz = c1.getSuperclass();
    String superClazzName = superClazz.getName();
    System.out.println("superClazzName = " + superClazzName);
    //獲取實現的接口
    Class[] interfaces = c1.getInterfaces();
    for (Class t : interfaces) {
    System.out.println("interfacesName = " + t.getName());
    }
    //獲取指定類的成員變量
    Field[] fields = c1.getDeclaredFields();
    for (Field field : fields) {
    modifier = Modifier.toString(field.getModifiers()); //獲取每個字段的訪問修飾符
    Class type = field.getType(); //獲取字段的數據類型所對應的Class對象
    String name = field.getName(); //獲取字段名
    if (type.isArray()) { //如果是數組類型則需要特別處理
    String arrType = type.getComponentType().getName() +
    "[]";
    System.out.println("" + modifier + " " + arrType + " "
    + name + ";");
    } else {
    System.out.println("" + modifier + " " + type + " " +
    name + ";");
    }
    }
    //獲取類的構造方法
    Constructor[] constructors = c1.getDeclaredConstructors();
    for (Constructor constructor : constructors) {
    String name = constructor.getName(); //構造方法名
    modifier = Modifier.toString(constructor.getModifiers()); //獲取訪問修飾符
    System.out.println("" + modifier +" " + name + "(");
    Class[] paramTypes = constructor.getParameterTypes(); //獲取構造方法中的參數
    for (int i = 0; i < paramTypes.length; i++) {
    if (i > 0) {
    System.out.print(",");
    }
    if (paramTypes[i].isArray()) {
    System.out.println(paramTypes
    [i].getComponentType().getName()+"[]");
    } else {
    System.out.print(paramTypes[i].getName());
    }
    }
    System.out.println(");");
    }
    //獲取成員方法
    Method[] methods = c1.getDeclaredMethods();
    for (Method method: methods) {
    modifier = Modifier.toString(method.getModifiers());
    Class returnType = method.getReturnType(); //獲取方法的返回類型
    if (returnType.isArray()) {
    String arrType = returnType.getComponentType
    ().getName()+"[]";
    System.out.print(""+modifier+" " + arrType + " " +
    method.getName() + "(");
    } else {
    System.out.print("" + modifier + " " +
    returnType.getName() + " " + method.getName() + "(");
    }
    Class[] paramTypes = method.getParameterTypes();
    for (int i = 0; i < paramTypes.length; i++) {
    if (i > 0) {
    System.out.print(",");
    }
    if (paramTypes[i].isArray()) {
    System.out.println(paramTypes
    [i].getComponentType().getName()+"[]");
    } else {
    System.out.print(paramTypes[i].getName());
    }
    }
    System.out.println(");");
    }

    TestReflection t1 = (TestReflection) c1.newInstance(); //利用反射來創建類的對象
    System.out.println("name == " + t1.name);
    System.out.println("learn == " + t1.learn);
    Method method = c1.getDeclaredMethod("setName", String.class);
    method.invoke(t1, "我是wblearn");
    System.out.println("name == " + t1.name);
    method = c1.getDeclaredMethod("setLearn", boolean.class);
    method.setAccessible(true);
    method.invoke(t1, true);
    System.out.println("learn == " + t1.learn);
    
    }

public String getName() {
    return name;
}
public void setName(String name) {
    this.name = name;
}
public boolean isLearn() {
    return learn;
}
public void setLearn(boolean learn) {
    this.learn = learn;
}
public int[] getAge() {
    return age;
}
public void setAge(int[] age) {
    this.age = age;
}

}

</pre>

打印結果:

這里寫圖片描述

6、反射的優缺點
優點:
(1)能夠運行時動態獲取類的實例,大大提高系統的靈活性和擴展性。
(2)與Java動態編譯相結合,可以實現無比強大的功能
缺點:
(1)使用反射的性能較低
(2)使用反射相對來說不安全
(3)破壞了類的封裝性,可以通過反射獲取這個類的私有方法和屬性

寫在最后

任何事物,都有兩面性,反射的優點,也同是就是它的缺點,所以,沒有好與壞,合適的場景應用才是最好的,正如前面說的,如果能有提供接口信息的池就用池的方式。這篇就簡單的回顧一下java反射機制,下篇就將反射應用到項目中將接口信息呈現在界面。另外,在學習編程的過程中,我覺得不止要獲得各種知識,更多的是通過學習技術知識提高解決問題的能力,這樣我們才能立于不敗之地!

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,363評論 6 532
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,497評論 3 416
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,305評論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,962評論 1 311
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,727評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,193評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,257評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,411評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,945評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,777評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,978評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,519評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,216評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,642評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,878評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,657評論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,960評論 2 373

推薦閱讀更多精彩內容

  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,782評論 18 139
  • 1. Java基礎部分 基礎部分的順序:基本語法,類相關的語法,內部類的語法,繼承相關的語法,異常的語法,線程的語...
    子非魚_t_閱讀 31,717評論 18 399
  • 從三月份找實習到現在,面了一些公司,掛了不少,但最終還是拿到小米、百度、阿里、京東、新浪、CVTE、樂視家的研發崗...
    時芥藍閱讀 42,319評論 11 349
  • 晚上吃完飯,我跟陽陽說:晚上幫媽媽洗碗啊?他看似不是很樂意,但還是答應了。因為他知道媽媽是真忙,明天要考試。...
    嫂子進城閱讀 625評論 1 2
  • 我云盤里一直放著賈樟柯的《天注定》,不過在高強度的代碼壓力下,我生活中的每一件事都在拖沓,所以到處都收藏了我沒去看...
    簡節蝴蝶閱讀 787評論 0 0