APT 系列 (一):APT 筑基之反射

相關(guān)文章:

Android APT 系列 (一):APT 筑基之反射
https://juejin.cn/post/6977679823132950536

什么是反射?

簡(jiǎn)單來(lái)講谒主,反射就是:已知一個(gè)類(lèi)朝扼,可以獲取這個(gè)類(lèi)的所有信息

一般情況下,根據(jù)面向?qū)ο蠓庋b原則霎肯,Java實(shí)體類(lèi)的屬性都是私有的擎颖,我們不能獲取類(lèi)中的屬性。但我們可以根據(jù)反射观游,獲取私有變量搂捧、方法、構(gòu)造方法懂缕,注解允跑,泛型等等,非常的強(qiáng)大

注意:GoogleAndroid 9.0及之后對(duì)反射做了限制搪柑,被使用 @hide標(biāo)記的屬性和方法通過(guò)反射拿不到

反射使用
//包路徑
package com.dream.aptdemo;

//自定義注解1
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface CustomAnnotation1{
  
}

//自定義注解2
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface CustomAnnotation2{
  
}

//自定義注解3
@Target(ElementType.TYPE)
@Inherited
@Retention(RetentionPolicy.RUNTIME)
@interface CustomAnnotation3{

}

//接口
interface ICar {
    void combine();
}

//車(chē)
@CustomAnnotation3
class Car<K,V> {
    private String carDesign = "設(shè)計(jì)稿";
    public String engine = "發(fā)動(dòng)機(jī)";

    public void run(long kilometer) {
        System.out.println("Car run " + kilometer + " km");
    }
}
//==============================上面這些都是為下面這臺(tái)奔馳服務(wù)的??===========================
//奔馳
@CustomAnnotation1
@CustomAnnotation2
class Benz extends Car<String,Integer> implements ICar {
  
    private String carName = "奔馳";
    public String carColor = "白色";
  
    public Benz() {
    }

    private Benz(String carName) {
        this.carName = carName;
    }

    public Benz(String carName, String carColor) {
        this.carName = carName;
        this.carColor = carColor;
    }

    @Override
    public void combine() {
        System.out.println("組裝一臺(tái)奔馳");
    }

    private void privateMethod(String params){
        System.out.println("我是私有方法: " + params);
    }
}
類(lèi)
獲取類(lèi)

3 種方式去獲取類(lèi)對(duì)象
1)Benz.class:類(lèi)獲取
2)benz.getClass:對(duì)象獲取
3)Class.forName:靜態(tài)獲取

 Benz benz = new Benz();
 Class benzClass = Benz.class;
 Class benzClass1 = benz.getClass();
 Class benzClass2 = Class.forName("com.dream.aptdemo.Benz");

注意:
1聋丝、在一個(gè)JVM中,一種類(lèi)拌屏,只會(huì)有一個(gè)類(lèi)對(duì)象存在潮针。所以以上三種方式取出來(lái)的類(lèi)對(duì)象,都是一樣的倚喂。
2、無(wú)論哪種途徑獲取類(lèi)對(duì)象瓣戚,都會(huì)導(dǎo)致靜態(tài)屬性被初始化端圈,而且只會(huì)執(zhí)行一次。(除了直接使用Benz.class類(lèi)獲取這種方式子库,這種方式不會(huì)導(dǎo)致靜態(tài)屬性被初始化)

獲取類(lèi)名
String className = benzClass.getSimpleName();
System.out.println(className);

//打印結(jié)果
Benz
獲取類(lèi)路徑
String classPath1 = benzClass.getName();
String classPath2 = benzClass.getCanonicalName();
System.out.println(classPath1);
System.out.println(classPath2);

//打印結(jié)果
com.pf.reflect.Benz
com.pf.reflect.Benz

這里可能大家會(huì)有個(gè)疑問(wèn):benzClass.getName()benzClass.getCanonicalName()有啥區(qū)別嗎舱权?
從上面打印結(jié)果來(lái)看,沒(méi)啥區(qū)別仑嗅,但是如果我們?cè)?code>Benz這個(gè)里面加個(gè)內(nèi)部類(lèi)宴倍,然后獲取內(nèi)部類(lèi)的路徑,你就會(huì)看到區(qū)別了:

//...
class Benz extends Car implements ICar {
    //...
    class InnerClass{
        
    }
}

Class<Benz.InnerClass> innerClass = Benz.InnerClass.class;
System.out.println(innerClass.getName());
System.out.println(innerClass.getCanonicalName());
//打印結(jié)果
com.pf.reflect.Benz$InnerClass
com.pf.reflect.Benz.InnerClass
獲取父類(lèi)名
String fatherClassName = benzClass.getSuperclass().getSimpleName();
System.out.println(fatherClassName);

//打印結(jié)果
Car
獲取接口
Class[] interfaces = benzClass.getInterfaces();
for (Class anInterface : interfaces) {
    System.out.println(anInterface.getName());
}

//打印結(jié)果
com.pf.reflect.inter.ICar
創(chuàng)建實(shí)例對(duì)象
//獲取構(gòu)造方法
Constructor constructor = benzClass.getDeclaredConstructor();
//創(chuàng)建實(shí)例
Benz myBenz = (Benz) constructor.newInstance();
//修改屬性
myBenz.carColor = "黑色";
myBenz.combine();
System.out.println(myBenz.carColor);

//打印結(jié)果
組裝一臺(tái)奔馳
黑色

注意:下面要講的關(guān)于帶Declare 的屬性和方法和不帶Declare 區(qū)別:
1仓技、帶Declare 的屬性和方法獲取的是本類(lèi)所有的屬性和方法鸵贬,不包含繼承得來(lái)的
2、不帶Declare的屬性和方法獲取的是所有public修飾的屬性和方法脖捻,包含繼承得來(lái)的
3阔逼、訪問(wèn)private 修飾的屬性和方法,需調(diào)用 setAccessible設(shè)置為true地沮,表示允許我們?cè)L問(wèn)私有變量

屬性
獲取單個(gè)屬性
Field carName = benzClass.getDeclaredField("carName");
獲取多個(gè)屬性
//獲取本類(lèi)全部屬性
Field[] declaredFields = benzClass.getDeclaredFields();
for (Field declaredField : declaredFields) {
    System.out.println("屬性: " + declaredField.getName());
}

//打印結(jié)果
屬性: carName
屬性: carColor

//獲取本類(lèi)及父類(lèi)全部 public 修飾的屬性
Field[] fields = benzClass.getFields();
for (Field field : fields) {
    System.out.println("屬性: " + field.getName());
}

//打印結(jié)果
屬性: carColor
屬性: engine
設(shè)置允許訪問(wèn)私有變量
carName.setAccessible(true);
獲取屬性名
System.out.println(carName.getName());

//打印結(jié)果
carName
獲取變量類(lèi)型
System.out.println(carName.getType().getName());

//打印結(jié)果
java.lang.String
獲取對(duì)象中該屬性的值
System.out.println(carName.get(benz));

//打印結(jié)果
奔馳
給屬性設(shè)置值
carName.set(benz,"sweetying");
System.out.println(carName.get(benz));

//打印結(jié)果
sweetying
方法
獲取單個(gè)方法
//獲取 public 方法
Method publicMethod = benzClass.getMethod("combine");

//獲取 private 方法
Method privateMethod = benzClass.getDeclaredMethod("privateMethod",String.class);
獲取多個(gè)方法
//獲取本類(lèi)全部方法
Method[] declaredMethods = benzClass.getDeclaredMethods();
for (Method declaredMethod : declaredMethods) {
    System.out.println("方法名: " + declaredMethod.getName());
}

//打印結(jié)果
方法名: privateMethod
方法名: combine


//獲取本類(lèi)及父類(lèi)全部 public 修飾的方法
Method[] methods = benzClass.getMethods();
for (Method method : methods) {
    System.out.println("方法名: " + method.getName());
}

//打印結(jié)果 因?yàn)樗蓄?lèi)默認(rèn)繼承 Object , 所以打印了 Object 的一些方法
方法名: combine
方法名: run
方法名: wait
方法名: wait
方法名: wait
方法名: equals
方法名: toString
方法名: hashCode
方法名: getClass
方法名: notify
方法名: notifyAll
方法調(diào)用
Method privateMethod = benzClass.getDeclaredMethod("privateMethod",String.class);
privateMethod.setAccessible(true);
privateMethod.invoke(benz,"接收傳入的參數(shù)");

//打印結(jié)果
我是私有方法: 接收傳入的參數(shù)
構(gòu)造方法
獲取單個(gè)構(gòu)造方法
//獲取本類(lèi)單個(gè)構(gòu)造方法
Constructor declaredConstructor = benzClass.getDeclaredConstructor(String.class);

//獲取本類(lèi)單個(gè) public 修飾的構(gòu)造方法
Constructor singleConstructor = benzClass.getConstructor(String.class,String.class);
獲取多個(gè)構(gòu)造方法
//獲取本類(lèi)全部構(gòu)造方法
Constructor[] declaredConstructors = benzClass.getDeclaredConstructors();
for (Constructor declaredConstructor1 : declaredConstructors) {
    System.out.println("構(gòu)造方法: " + declaredConstructor1);
}
//打印結(jié)果
構(gòu)造方法: public com.dream.aptdemo.Benz()
構(gòu)造方法: public com.dream.aptdemo.Benz(java.lang.String,java.lang.String)
構(gòu)造方法: private com.dream.aptdemo.Benz(java.lang.String)


//獲取全部 public 構(gòu)造方法, 不包含父類(lèi)的構(gòu)造方法
Constructor[] constructors = benzClass.getConstructors();
for (Constructor constructor1 : constructors) {
    System.out.println("構(gòu)造方法: " + constructor1);
}
//打印結(jié)果
構(gòu)造方法: public com.dream.aptdemo.Benz()
構(gòu)造方法: public com.dream.aptdemo.Benz(java.lang.String,java.lang.String)
構(gòu)造方法實(shí)例化對(duì)象
//以上面 declaredConstructor 為例
declaredConstructor.setAccessible(true);
Benz declareBenz = (Benz) declaredConstructor.newInstance("");
System.out.println(declareBenz.carColor);
//打印結(jié)果
白色

//以上面 singleConstructor 為例
Benz singleBenz = (Benz) singleConstructor.newInstance("奔馳 S ","香檳金");
System.out.println(singleBenz.carColor);
//打印結(jié)果
香檳金
泛型
獲取父類(lèi)的泛型
Type genericType = benzClass.getGenericSuperclass();
if (genericType instanceof ParameterizedType) {
   Type[] actualType = ((ParameterizedType) genericType).getActualTypeArguments();
   for (Type type : actualType) {
       System.out.println(type.getTypeName());
   }
}
//打印結(jié)果
java.lang.String
java.lang.Integer
注解
獲取單個(gè)注解
//獲取單個(gè)本類(lèi)或父類(lèi)注解
Annotation annotation1 = benzClass.getAnnotation(CustomAnnotation1.class);
System.out.println(annotation1.annotationType().getSimpleName());
Annotation annotation3 = benzClass.getAnnotation(CustomAnnotation3.class);
System.out.println(annotation3.annotationType().getSimpleName());
//打印結(jié)果
CustomAnnotation1
CustomAnnotation3

//獲取單個(gè)本類(lèi)注解
Annotation declaredAnnotation1 = benzClass.getDeclaredAnnotation(CustomAnnotation2.class);
System.out.println(declaredAnnotation1.annotationType().getSimpleName());
//打印結(jié)果
CustomAnnotation2
獲取全部注解
//獲取本類(lèi)和父類(lèi)的注解(父類(lèi)的注解需用 @Inherited 表示可被繼承)
Annotation[] annotations = benzClass.getAnnotations();
for (Annotation annotation : annotations) {
    System.out.println("注解名稱(chēng): " + annotation.annotationType().getSimpleName());
}
//打印結(jié)果
注解名稱(chēng): CustomAnnotation3
注解名稱(chēng): CustomAnnotation1
注解名稱(chēng): CustomAnnotation2

//獲取本類(lèi)的注解
Annotation[] declaredAnnotations = benzClass.getDeclaredAnnotations();
for (Annotation declaredAnnotation : declaredAnnotations) {
    System.out.println("注解名稱(chēng): " + declaredAnnotation.annotationType().getSimpleName());
}
//打印結(jié)果
注解名稱(chēng): CustomAnnotation1
注解名稱(chēng): CustomAnnotation2
反射實(shí)踐

需求大概就是:通過(guò)后臺(tái)配置下發(fā)嗜浮,完成 App 業(yè)務(wù)功能的切換羡亩。因?yàn)橹皇悄M,我們這里就以通過(guò)讀取本地配置文件完成 App 業(yè)務(wù)功能的切換:

首先準(zhǔn)備兩個(gè)業(yè)務(wù)類(lèi)危融,假設(shè)他們的功能都很復(fù)雜
//包名
package com.dream.aptdemo;

//業(yè)務(wù)1
class Business1 {
 
    public void doBusiness1Function(){
        System.out.println("復(fù)雜業(yè)務(wù)功能1");
    }
}

//業(yè)務(wù)2
class Business2 {
 
    public void doBusiness2Function(){
        System.out.println("復(fù)雜業(yè)務(wù)功能2");
    }
}
非反射方式
public class Client {
  
    @Test
    public void test() {
        //業(yè)務(wù)功能1
        new Business1().doBusiness1Function();
    }
}

假設(shè)這個(gè)時(shí)候需要從第一個(gè)業(yè)務(wù)功能切換到第二個(gè)業(yè)務(wù)功能畏铆,使用非反射方式,必須修改代碼吉殃,并且重新編譯運(yùn)行及志,才可以達(dá)到效果。那么我們可以通過(guò)反射去通過(guò)讀取配置從而完成功能的切換寨腔,這樣我們就不需要修改代碼且代碼變得更加通用

反射方式

首先準(zhǔn)備一個(gè)配置文件速侈,如下圖:


配置文件截圖.png

讀取配置文件,反射創(chuàng)建實(shí)例并調(diào)用方法

public class Client {
  
    @Test
    public void test() throws Exception {
        try {
            //獲取文件
            File springConfigFile = new File("/Users/zhouying/AndroidStudioProjects/AptDemo/config.txt");
            //讀取配置
            Properties config= new Properties();
            config.load(new FileInputStream(springConfigFile));
            //獲取類(lèi)路徑
            String classPath = (String) config.get("class");
            //獲取方法名
            String methodName = (String) config.get("method");
                        
            //反射創(chuàng)建實(shí)例并調(diào)用方法
            Class aClass = Class.forName(classPath);
            Constructor declaredConstructor = aClass.getDeclaredConstructor();
            Object o = declaredConstructor.newInstance();
            Method declaredMethod = aClass.getDeclaredMethod(methodName);
            declaredMethod.invoke(o);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

完成上面兩步后迫卢,后續(xù)我們就只需要修改配置文件就能完成 App 業(yè)務(wù)功能的切換了

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末倚搬,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子乾蛤,更是在濱河造成了極大的恐慌每界,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,122評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件家卖,死亡現(xiàn)場(chǎng)離奇詭異眨层,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)上荡,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,070評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)趴樱,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人酪捡,你說(shuō)我怎么就攤上這事叁征。” “怎么了逛薇?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,491評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵捺疼,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我永罚,道長(zhǎng)啤呼,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,636評(píng)論 1 293
  • 正文 為了忘掉前任呢袱,我火速辦了婚禮官扣,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘产捞。我一直安慰自己醇锚,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,676評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著焊唬,像睡著了一般恋昼。 火紅的嫁衣襯著肌膚如雪赶促。 梳的紋絲不亂的頭發(fā)上液肌,一...
    開(kāi)封第一講書(shū)人閱讀 51,541評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音鸥滨,去河邊找鬼嗦哆。 笑死,一個(gè)胖子當(dāng)著我的面吹牛婿滓,可吹牛的內(nèi)容都是我干的老速。 我是一名探鬼主播,決...
    沈念sama閱讀 40,292評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼凸主,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼橘券!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起卿吐,我...
    開(kāi)封第一講書(shū)人閱讀 39,211評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤旁舰,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后嗡官,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體箭窜,經(jīng)...
    沈念sama閱讀 45,655評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,846評(píng)論 3 336
  • 正文 我和宋清朗相戀三年衍腥,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了磺樱。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,965評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡紧阔,死狀恐怖坊罢,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情擅耽,我是刑警寧澤,帶...
    沈念sama閱讀 35,684評(píng)論 5 347
  • 正文 年R本政府宣布物遇,位于F島的核電站乖仇,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏询兴。R本人自食惡果不足惜乃沙,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,295評(píng)論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望诗舰。 院中可真熱鬧警儒,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,894評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至记劝,卻和暖如春变姨,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背厌丑。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,012評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工定欧, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人怒竿。 一個(gè)月前我還...
    沈念sama閱讀 48,126評(píng)論 3 370
  • 正文 我出身青樓砍鸠,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親耕驰。 傳聞我的和親對(duì)象是個(gè)殘疾皇子爷辱,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,914評(píng)論 2 355

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