Java高級(jí)語言特性之注解

注解的定義

Java 注解(Annotation)又稱 Java 標(biāo)注,是 JDK1.5 引入的一種注釋機(jī)制卖鲤。

注解是元數(shù)據(jù)的一種形式肾扰,提供有關(guān)于程序但不屬于程序本身的數(shù)據(jù)。注解對(duì)它們注解的代碼的操作沒有直接影響蛋逾。
注解本身沒有任何意義集晚,單獨(dú)的注解就是一種注釋,他需要結(jié)合其他如反射区匣、插樁等技術(shù)才有意義偷拔。

如何定義一個(gè)注解

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.SOURCE)
public @interface Example {
    String value() default "xxx";
}

這里是注解的一個(gè)簡(jiǎn)單的例子,在接口前面加上一個(gè)@,就能定義一個(gè)注解了莲绰。
在這個(gè)注解中還有一個(gè)value()的成員變量欺旧,其中我們?yōu)樗x了默認(rèn)值“xxx”,如果沒有默認(rèn)值蛤签,那么該注解使用的時(shí)候辞友,就必須為它傳值。

注解上面還有兩個(gè)注解震肮,我們將之稱為元注解称龙。

元注解

元注解,即在定義注解時(shí)戳晌,注解類也能夠使用其他的注解聲明鲫尊。這種對(duì)注解類型進(jìn)行注解的注解類,我們稱之為 meta-annotation(元注解)沦偎。

聲明的注解允許作用于哪些節(jié)點(diǎn)使用@Target聲明疫向,例如ElementType.FIELD允許在成員變量上使用,而@Target注解是一個(gè)一對(duì)多的關(guān)系扛施,即我們所寫的注解可以在多個(gè)地方定義鸿捧,在類上定義,在方法上定義疙渣,等等匙奴;

保留級(jí)別由@Retention 聲明。其中保留級(jí)別如下妄荔。

  • RetentionPolicy.SOURCE
  • 標(biāo)記的注解僅保留在源級(jí)別中泼菌,并被編譯器忽略。
  • RetentionPolicy.CLASS
  • 標(biāo)記的注解在編譯時(shí)由編譯器保留啦租,但 Java 虛擬機(jī)(JVM)會(huì)忽略哗伯。
  • RetentionPolicy.RUNTIME
  • 標(biāo)記的注解由 JVM 保留,因此運(yùn)行時(shí)環(huán)境可以使用它篷角。

當(dāng)我們使用@SOURCE聲明注解時(shí)候焊刹,@Example注解的保留級(jí)別為SOURSE,即保留到源碼階段恳蹲。

@Example("123")
public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
    }
}

通過ASM反編譯工具查看MainActivity.class字節(jié)碼虐块,并沒有看到注解的存在,因?yàn)樵诰幾g過程中嘉蕾,注解已經(jīng)被抹除了

// class version 51.0 (51)
// access flags 0x21
這里并沒有看到注解的存在了
public class com/example/anatationtest/MainActivity extends androidx/appcompat/app/AppCompatActivity {

  // compiled from: MainActivity.java

  // access flags 0x1
  public <init>()V
  ......
  protected onCreate(Landroid/os/Bundle;)V
   ......
}

注解的應(yīng)用場(chǎng)景

根據(jù)注解的保留級(jí)別不同贺奠,對(duì)注解的使用自然存在不同場(chǎng)景。由注解的三個(gè)不同保留級(jí)別可知错忱,注解作用于:
源碼儡率、字節(jié)碼與運(yùn)行時(shí)可以產(chǎn)生不同的應(yīng)用場(chǎng)景挂据。

級(jí)別 技術(shù) 說明
源碼 APT 在編譯期能夠獲取注解與注解聲明的類包括類中所有成員信息,一般用于生成額外的輔助類
字節(jié)碼 字節(jié)碼增強(qiáng) 在編譯出Class后儿普,通過修改Class數(shù)據(jù)以實(shí)現(xiàn)修改代碼邏輯目的崎逃。對(duì)于是否需要修改的區(qū)分或者修改為不同邏輯的判斷可以使用注解
運(yùn)行時(shí) 反射 在程序運(yùn)行期間,通過反射技術(shù)動(dòng)態(tài)獲取注解與其元素箕肃,從而完成不同的邏輯判定

APT技術(shù)

APT技術(shù)婚脱,APT今魔,全稱為Annotation Processor Tools 勺像,即注解處理器

要定義一個(gè)注解處理器,首先要新建一個(gè)普通Java模塊错森,并在app模塊中依賴吟宦。
在其中新建一個(gè)注解處理器,繼承自AbstractProcessor涩维,編譯器已經(jīng)為我們?cè)趦?nèi)部實(shí)現(xiàn)了注解的采集殃姓,我們只需要對(duì)注解進(jìn)行處理就可以了。

@SupportedAnnotationTypes("com.example.anatationtest.Example")
public class ExampleProcessor extends AbstractProcessor {
    @Override
    public boolean process(Set<? extends TypeElement> set, RoundEnvironment roundEnvironment) {
        Messager messager = processingEnv.getMessager();
        messager.printMessage(Diagnostic.Kind.NOTE,"========這里是打印信息========");
        return false;
    }
}

正如Activity類需要在Manifest中注冊(cè)一樣瓦阐,注解器也需要配置才可以生效蜗侈。配置文件層級(jí)為compiler/main/resources/META-INF/services/javax.annotation.processing.Processor

其中配置文件內(nèi)容為,即定義的注解處理器的路徑

com.example.compile.ExampleProcessor

注解處理程序運(yùn)行在什時(shí)候

我們知道睡蟋,一個(gè).java文件要由javac編譯成.class文件踏幻,并交由虛擬機(jī)去運(yùn)行。在這個(gè)過程中戳杀,javac會(huì)采集到所有的注解信息该面,并包裝成Element節(jié)點(diǎn),然后交給注解處理程序信卡。那么怎么證明這一點(diǎn)呢隔缀?

試著Make Project,可以在Build Output中的compileDebugJavaWithJavac Task中看到我們寫在代碼中的打印信息傍菇,說明javac編譯.java文件的階段調(diào)起了注解處理程序猾瘸。

Android注解語法檢查

在Android中我們需要設(shè)計(jì)接口以供使用者調(diào)用時(shí),如出現(xiàn)需要對(duì)入?yún)⑦M(jìn)行類型限定丢习,如限定為資源ID牵触、布局ID等類型參數(shù),將參數(shù)類型直接給定int即可泛领。然而荒吏,我們可以利用Android為我們提供的語法檢查注解,來輔助進(jìn)行更為直接的參數(shù)類型檢查與提示渊鞋。

如參數(shù)限制為:圖片資源ID绰更。這里利用了@Drawable 來限定入?yún)镈rawable類型的int值

public Drawable getMyDrawable(@DrawableRes int id) {
       return getDrawable(id);
}

在平時(shí)開發(fā)中假如有一個(gè)方法限制了入?yún)⒌念愋颓萍罚敲次覀兛梢允褂妹杜e來解決。

private Weekday currentDay;

    enum Weekday {
        SUNDAY,MONDAY
    }
    public void setCurrentDay(Weekday currentDay){
        this.currentDay=currentDay;
}

但是通過ASM字節(jié)碼工具可以發(fā)現(xiàn)儡湾,枚舉其實(shí)是生成了對(duì)象特恬,較int基本數(shù)據(jù)類型會(huì)比較占用內(nèi)存

// access flags 0x4019
public final static enum Lcom/enjoy/ annotat ion/ intdef/Test$WeekDay; SUNDAY
// access flags 0x4019
public final static enum Lcom/ enjoy/ annotation/ intdef /Test$WeekDay; MONDAY

這時(shí)候就可以使用@IntDef注解來進(jìn)行語法檢查,@IntDef是AndroidX為我們提供的一個(gè)元注解徐钠。由IDE來實(shí)現(xiàn)癌刽,在我們編寫代碼的時(shí)候進(jìn)行檢查。

@WekDay
private static int mCurrentIntDay;

@IntDef({SUNDAY, MONDAY})
@Target({ElementType.FIELD, ElementType.PARAMETER})
@Retention(RetentionPolicy.SOURCE)
@interface WekDay {  //注解

}

public static void setCurrentDay(@WekDay int currentDay) {
   mCurrentIntDay = currentDay;
}

但是尝丐,語法檢查階段對(duì)于我們編譯是不會(huì)產(chǎn)生影響的显拜。

字節(jié)碼增強(qiáng)技術(shù)

什么叫字節(jié)碼增強(qiáng)技術(shù)?就是在字節(jié)碼中寫代碼爹袁。平時(shí)我們是在.java文件中去編寫代碼远荠,其有一定的格式,也正如.java一樣失息,.class文件也有一定的格式(數(shù)據(jù)按照特定的方式記錄與排列)譬淳。

QQ空間曾經(jīng)發(fā)布的熱修復(fù)解決方案中利用Javaassist 庫(kù)實(shí)現(xiàn)向類的構(gòu)造函數(shù)中插入一段代碼解決
CLASS_ISPREVERIFIED 問題。包括了Instant Run的實(shí)現(xiàn)以及參照Instant Run實(shí)現(xiàn)的熱修復(fù)美團(tuán)Robus等等等等都利用到了插樁技術(shù)盹兢。

插樁就是將一段代碼插入到另一段代碼邻梆,或替換另一段代碼。字節(jié)碼插樁顧名思義就是在我們編寫的源碼編譯成字節(jié)碼(Class)后绎秒,在Android下生成dex之前修改Class文件浦妄,修改或者增強(qiáng)原有代碼邏輯的操作。

由于對(duì)于該技術(shù)也只是有所了解替裆,因此不做更多贅述校辩。如果大家感興趣可以自己去加強(qiáng)學(xué)習(xí)。

利用注解加反射實(shí)現(xiàn)findViewById

我們利用注解加反射來實(shí)現(xiàn)一個(gè)簡(jiǎn)單的findViewById辆童。

首先我們定義一個(gè)@InjectView的注解 宜咒,里面包含一個(gè)@Idres int類型的成員變量,用來存放控件的id值把鉴。
由于程序需要在運(yùn)行期間利用反射來獲取元素的注解和值故黑,因此注解應(yīng)聲明在Runtime階段執(zhí)行

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface InjectView {
    @IdRes int value();
}

使用注解:

public class MainActivity extends AppCompatActivity {

    @InjectView(R.id.tv_text)
    TextView textView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        InjectUtil.injectView(this);
        textView.setText("使用了@InjectView注解");
    }
}

InjectViewUtil 處理工具類

public class InjectUtil {
    public static void injectView(Activity activity) {
        Class<? extends Activity> cls = activity.getClass();
        //獲得成員變量
        Field[] declaredFields = cls.getDeclaredFields();
        for (Field declaredField : declaredFields) {
            //判斷是否被@Inject注解
            if (declaredField.isAnnotationPresent(InjectView.class)) {
                InjectView annotation = declaredField.getAnnotation(InjectView.class);
                //獲得注解的值
                int id = annotation.value();
                View view = activity.findViewById(id);
                //反射設(shè)置屬性的值
                declaredField.setAccessible(true);//設(shè)置訪問權(quán)限,允許操作private屬性
                try {
                    declaredField.set(activity, view);
                } catch (IllegalAccessException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

InjectViewUtil在處理時(shí)庭砍,先通過getClass拿到Activity的類對(duì)象场晶,再使用getDeclaredFields拿到其成員變量,并通過if (declaredField.isAnnotationPresent(InjectView.class))
判斷是否被@InjectView注解過了怠缸,并進(jìn)一步拿到id值诗轻,最后使用set方法設(shè)置回去。運(yùn)行項(xiàng)目揭北,觀察效果扳炬,TextView 對(duì)象已經(jīng)獲取到了實(shí)例吏颖,并修改為了我們?cè)O(shè)置的text值。

這是ButterKnife早期的實(shí)現(xiàn)恨樟,但由于運(yùn)行階段利用反射去處理注解半醉,會(huì)影響運(yùn)行時(shí)的性能,所以后面它是在編譯時(shí)對(duì)注解進(jìn)行解析完成相關(guān)代碼的生成劝术,即剛剛介紹的第一種缩多,利用注解處理器去完成findViewById的過程,相關(guān)源碼大家感興趣的話可以去查閱养晋。

關(guān)于java注解的知識(shí)本次就介紹到這~

じゃ衬吆、また

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市匙握,隨后出現(xiàn)的幾起案子咆槽,更是在濱河造成了極大的恐慌,老刑警劉巖圈纺,帶你破解...
    沈念sama閱讀 206,968評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異麦射,居然都是意外死亡蛾娶,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門潜秋,熙熙樓的掌柜王于貴愁眉苦臉地迎上來蛔琅,“玉大人,你說我怎么就攤上這事峻呛÷奘郏” “怎么了?”我有些...
    開封第一講書人閱讀 153,220評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵钩述,是天一觀的道長(zhǎng)寨躁。 經(jīng)常有香客問我,道長(zhǎng)牙勘,這世上最難降的妖魔是什么职恳? 我笑而不...
    開封第一講書人閱讀 55,416評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮方面,結(jié)果婚禮上放钦,老公的妹妹穿的比我還像新娘。我一直安慰自己恭金,他們只是感情好操禀,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評(píng)論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著横腿,像睡著了一般颓屑。 火紅的嫁衣襯著肌膚如雪辙培。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,144評(píng)論 1 285
  • 那天邢锯,我揣著相機(jī)與錄音扬蕊,去河邊找鬼。 笑死丹擎,一個(gè)胖子當(dāng)著我的面吹牛尾抑,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播蒂培,決...
    沈念sama閱讀 38,432評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼再愈,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了护戳?” 一聲冷哼從身側(cè)響起翎冲,我...
    開封第一講書人閱讀 37,088評(píng)論 0 261
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎媳荒,沒想到半個(gè)月后抗悍,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,586評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡钳枕,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評(píng)論 2 325
  • 正文 我和宋清朗相戀三年缴渊,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片鱼炒。...
    茶點(diǎn)故事閱讀 38,137評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡衔沼,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出昔瞧,到底是詐尸還是另有隱情指蚁,我是刑警寧澤,帶...
    沈念sama閱讀 33,783評(píng)論 4 324
  • 正文 年R本政府宣布自晰,位于F島的核電站凝化,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏缀磕。R本人自食惡果不足惜缘圈,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望袜蚕。 院中可真熱鬧糟把,春花似錦、人聲如沸牲剃。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽凿傅。三九已至缠犀,卻和暖如春数苫,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背辨液。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工虐急, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人滔迈。 一個(gè)月前我還...
    沈念sama閱讀 45,595評(píng)論 2 355
  • 正文 我出身青樓止吁,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親燎悍。 傳聞我的和親對(duì)象是個(gè)殘疾皇子敬惦,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評(píng)論 2 345

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