亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

自定義Android注解系列教程之注解變量

 更新時間:2018年07月06日 15:09:28   作者:idisfkj  
這篇文章主要給大家介紹了關(guān)于自定義Android注解系列教程之注解變量的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

前言

對于Android注解,或多或少都有一點接觸,但相信大多數(shù)人都是在使用其它依賴庫的時候接觸的。因為有些庫如果你想使用它就必須使用它所提供的注解。例如:ButterKnife、Dagger2、Room等等。

至于為何使用注解?使用過的應(yīng)該都知道,最明顯的就是方便、簡潔。通過使用注解可以在項目編譯階段,幫助我們自動生成一些重復(fù)的代碼,減輕我們的負擔(dān)。典型的ButterKnife本質(zhì)就是使用Android注解,通過注解來減少我們對view.findViewById的編寫,提高我們的開發(fā)效率。上一個系列(AAC)的Room也是一樣,我們可以簡單的回顧一下:

@Entity(tableName = "contacts")
data class ContactsModel(
 @PrimaryKey
 @ColumnInfo(name = "contacts_id")
 val id: Int,
 @ColumnInfo(name = "name")
 val name: String,
 @ColumnInfo(name = "phone")
 val phone: String
)

通過使用注解來定義一個實體表,也就10行左右的代碼。如果要我們?nèi)孔约簩懩墙^對要兩三百行代碼了,而且其中還可能出錯,又要改bug等等。效率就嚴重降低。對于依賴庫如果都這么麻煩也就不會有人用了。

那么如何判斷一個依賴庫是否需要使用注解呢?其實很簡單,只要記住以下兩點即可:

  • 需要生成的代碼不能與項目邏輯有關(guān)
  • Android注解只能生成代碼,并不能修改代碼

這里透露一下,Android注解的本質(zhì)是使用Java的反射機制,后續(xù)會詳細說明

項目架構(gòu)

相信ButterKnife應(yīng)該有接觸過吧,沒有的也沒關(guān)系,現(xiàn)在正是時候。下面我們會自己實現(xiàn)BindView與OnClick注解,實現(xiàn)ButterKnife中的對應(yīng)注解功能。那么我先來看下整體的項目架構(gòu)

通過項目圖,我們可以清晰的看到,主要分為三個部分

  • butterknife-annotations:注解庫,包含BindView與OnClick等自定義的注解
  • butterknife-bind:綁定庫,自定義的注解與聲明的類綁定
  • butterknife-compiler: 解析編譯生成庫,解析聲明類中的注解,在編譯時自動生成相應(yīng)的代碼。

為了幫助大家能夠更輕松的理解Android注解,今天主要分析的就是butterknife-annotations這個注解庫。帶大家一起來聲明注解變量。

BindView

為了要實現(xiàn)開源庫butterknife類似的綁定id效果,這里我們先定義一個BindView注解,具體如下:

@Retention(RetentionPolicy.SOURCE)
@Target(ElementType.FIELD)
public @interface BindView {
 @IdRes int[] value();
}

嗯,還是很簡單的對吧。也就5行代碼解決BindView注解的定義。

那么再來詳細剖析這5行代碼。

Retention

首先是第一行代碼的Retention,看它的使用方式就能知道,它也是一個聲明了的注解。

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Retention {
 /**
 * Returns the retention policy.
 * @return the retention policy
 */
 RetentionPolicy value();
}

通過源碼我們可以看出該注解只接收一個參數(shù),該參數(shù)為RetentionPolicy類型。那么我們在進一步深入RetentionPolicy:

public enum RetentionPolicy {
 /**
 * Annotations are to be discarded by the compiler.
 */
 SOURCE,
 
 /**
 * Annotations are to be recorded in the class file by the compiler
 * but need not be retained by the VM at run time. This is the default
 * behavior.
 */
 CLASS,
 
 /**
 * Annotations are to be recorded in the class file by the compiler and
 * retained by the VM at run time, so they may be read reflectively.
 *
 * @see java.lang.reflect.AnnotatedElement
 */
 RUNTIME
}

在這里我們發(fā)現(xiàn)它其實是一個枚舉,在枚舉中支持三個常量,分別為SOURCE、CLASS與RUNTIME。它們的區(qū)別主要是作用的周期范圍,下面我再對這三個的作用進行翻譯一遍:

  • SOURCE: 使用該標(biāo)明的注解將在編譯階段就被拋棄掉。
  • CLASS:使用該標(biāo)明的注解將在編譯階段記錄到生成的class文件中,但在運行階段時又會被VM拋棄。默認是該模式。
  • RUNTIME:使用該標(biāo)明的注解將在編譯階段被保存在生成的class文件中,同時在運行階段時會保存到VM中。所以它該注解將一直存在,自然能夠通過java的反射機制進行讀取。

所以它們的存在的生命時長為SOURCE < CLASS < RUNTIME。知道了它的作用范圍之后,我們在自定義注解時就要盡量較小注解的作用范圍,提高項目的編譯與運行速度。

因為我們的BindView注解只是為了進行Viwe的綁定,所以在編譯之后就無需存在,所以這里就使用了CLASS來進行標(biāo)明。

Target

下面我們在來看第二行代碼,這里使用到了另一個注解Target,我們還是來看下它的源碼:

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.ANNOTATION_TYPE)
public @interface Target {
 /**
 * Returns an array of the kinds of elements an annotation type
 * can be applied to.
 * @return an array of the kinds of elements an annotation type
 * can be applied to
 */
 ElementType[] value();
}

可以看到注解的源碼都非常簡單,這里接收了一個ElementType數(shù)組參數(shù),ElementType不難猜出它的類型也是一個枚舉:

public enum ElementType {
 /** Class, interface (including annotation type), or enum declaration */
 TYPE,
 
 /** Field declaration (includes enum constants) */
 FIELD,
 
 /** Method declaration */
 METHOD,
 
 /** Formal parameter declaration */
 PARAMETER,
 
 /** Constructor declaration */
 CONSTRUCTOR,
 
 /** Local variable declaration */
 LOCAL_VARIABLE,
 
 /** Annotation type declaration */
 ANNOTATION_TYPE,
 
 /** Package declaration */
 PACKAGE,
 
 /**
 * Type parameter declaration
 *
 * @since 1.8
 */
 TYPE_PARAMETER,
 
 /**
 * Use of a type
 *
 * @since 1.8
 */
 TYPE_USE
}

ElementType中雖然有10常量,但我們實際真正常用的也就是前面8種。它們代表自定義的注解能夠作用的對象。分別為:

  • TYPE: 作用于類、接口或者枚舉
  • FIELD:作用于類中聲明的字段或者枚舉中的常量
  • METHOD:作用于方法的聲明語句中
  • PARAMETER:作用于參數(shù)聲明語句中
  • CONSTRUCTOR:作用于構(gòu)造函數(shù)的聲明語句中
  • LOCAL_VARIABLE:作用于局部變量的聲明語句中
  • ANNOTATION_TYPE:作用于注解的聲明語句中
  • PACKAGE:作用于包的聲明語句中
  • TYPE_PARAMETER:java 1.8之后,作用于類型聲明的語句中
  • TYPE_USE:java 1.8之后,作用于使用類型的任意語句中

結(jié)合我們的BindView的作用是對View進行id綁定,自然是作用與聲明的字段上。所以在BindView中使用了FIELD。

再來看第四行代碼

@IdRes int[] value()

有了上面的注解接觸,不難理解這是標(biāo)明BindView將接收一個int類型的數(shù)組參數(shù)。對于開源庫butterknife中的BindView是接收需要綁定的View的id,這里我們做一個改版,再接收一個String的id,用來為綁定的View設(shè)置默認值。這樣我們自定義了的BindView注釋就完成了。

OnClick

下面我們再自定義一個OnClick點擊的注解,經(jīng)過上面的分析,可以在腦海中想想Retention與Target分別什么值?

想好了之后我們在來過一遍

@Retention(RetentionPolicy.SOURCE)
@Target(ElementType.METHOD)
public @interface OnClick {
 @IdRes int value();
}

Retention的作用范圍與BindView一樣首頁SOURCE,在編譯之后就無需存在;Target的作用對象與BindView不同,既然是點擊事件的點擊操作,自然是作用在操作邏輯的方法上,所以這里使用METHOD。

keep

文章開頭有提及到本質(zhì)是通過注解來自動生成代碼,為我們創(chuàng)建所需的類,那么在實際開發(fā)中一旦我們的項目混淆了,這將會導(dǎo)致自動創(chuàng)建的類失效,從而導(dǎo)致我們自定義的注解失效。所以為了防止其失效,我們在這里再定義一個注解keep:

@Retention(RetentionPolicy.CLASS)
@Target(ElementType.TYPE)
public @interface Keep {
}

Retention的作用范圍是在class文件中還要能夠被其它class調(diào)用,所以這里使用CLASS;Target作用對象是自動生成的類,所以使用TYPE。至于參數(shù)則不必要,它只是為了標(biāo)明類,防止其被混淆。

總結(jié)

butterknife-annotations庫中的自定義注解就完成了。通過上面的分析,我們注意點主要歸結(jié)于以下三點:

  • Retention: 明確注解作用的生命時長,盡早的消除
  • Target: 明確注解作用的對象
  • Keep: 防止后續(xù)自動生成的類被混淆

注解變量的定義就到這結(jié)束了,同時文章中的代碼都可以在Github (本地下載)中獲取到。使用時請將分支切換到feat_annotation_processing

好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

相關(guān)文章

  • JetpackCompose Navigation導(dǎo)航實現(xiàn)流程

    JetpackCompose Navigation導(dǎo)航實現(xiàn)流程

    Navigation是Jetpack用于Android導(dǎo)航的組件,作用是處理頁面跳轉(zhuǎn),以及頁面跳轉(zhuǎn)過程中的交互。使用Navigation,你就需要為每個頁面設(shè)定一條唯一路徑,它是一個String常量,形式是DeepLink的樣子,從一個頁面跳轉(zhuǎn)到另一個頁面,它通過輸入目的地的路徑進行轉(zhuǎn)跳
    2023-01-01
  • 在Android上實現(xiàn)HttpServer的示例代碼

    在Android上實現(xiàn)HttpServer的示例代碼

    本篇文章主要介紹了在Android上實現(xiàn)HttpServer的示例代碼,實現(xiàn)Android本地的微型服務(wù)器,具有一定的參考價值,有興趣的可以了解一下
    2017-08-08
  • Android實現(xiàn)TCP客戶端接收數(shù)據(jù)的方法

    Android實現(xiàn)TCP客戶端接收數(shù)據(jù)的方法

    這篇文章主要介紹了Android實現(xiàn)TCP客戶端接收數(shù)據(jù)的方法,較為詳細的分析了Android基于TCP實現(xiàn)客戶端接收數(shù)據(jù)的相關(guān)技巧與注意事項,需要的朋友可以參考下
    2016-04-04
  • Android快速實現(xiàn)無預(yù)覽拍照功能

    Android快速實現(xiàn)無預(yù)覽拍照功能

    這篇文章主要為大家詳細介紹了Android快速實現(xiàn)無預(yù)覽拍照功能,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-06-06
  • Android不規(guī)則封閉區(qū)域填充色彩的實例代碼

    Android不規(guī)則封閉區(qū)域填充色彩的實例代碼

    這篇文章主要介紹了Android不規(guī)則封閉區(qū)域填充色彩的實例代碼, 具有很好的參考價值,希望對大家有所幫助,一起跟隨小編過來看看吧
    2018-05-05
  • android利用ContentResolver訪問者獲取手機短信信息

    android利用ContentResolver訪問者獲取手機短信信息

    本篇文章主要介紹了android利用ContentResolver訪問者獲取手機短信信息,具有一定的參考價值,感興趣的小伙伴們可以參考一下。
    2017-02-02
  • Android實現(xiàn)水波紋點擊效果

    Android實現(xiàn)水波紋點擊效果

    這篇文章主要為大家詳細介紹了Android實現(xiàn)水波紋點擊效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-03-03
  • Android?Retrofit使用詳細教程

    Android?Retrofit使用詳細教程

    Retrofit是Android用來接口請求的網(wǎng)絡(luò)框架,內(nèi)部是基于OkHttp實現(xiàn)的,retrofit負責(zé)接口請求的封裝,retrofit可以直接將接口數(shù)據(jù)解析為Bean類、List集合等,直接簡化了中間繁瑣的數(shù)據(jù)解析過程,這篇文章主要介紹了Android?Retrofit使用詳情,需要的朋友可以參考下
    2024-03-03
  • kotlin 協(xié)程上下文異常處理詳解

    kotlin 協(xié)程上下文異常處理詳解

    這篇文章主要為大家介紹了kotlin 協(xié)程上下文異常處理詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-08-08
  • MacBook M1 Flutter環(huán)境搭建的實現(xiàn)步驟

    MacBook M1 Flutter環(huán)境搭建的實現(xiàn)步驟

    本文主要介紹了MacBook M1 Flutter環(huán)境搭建,F(xiàn)lutter官方和各項配套的軟件環(huán)境也還沒有成熟,導(dǎo)致搭建環(huán)境時碰到了不少坑,本文就詳細的介紹一下
    2021-08-08

最新評論