java高級語言特性,Java高級語言特性之注解

注解的定義

Java 注解(Annotation)又稱 Java 標注,是 JDK1.5 引入的一種注釋機制。

注解是元數據的一種形式,提供有關于程序但不屬于程序本身的數據。注解對它們注解的代碼的操作沒有直接影響。

注解本身沒有任何意義,單獨的注解就是一種注釋,他需要結合其他如反射、插樁等技術才有意義。

如何定義一個注解

@Target({ElementType.TYPE})

@Retention(RetentionPolicy.SOURCE)

public @interface Example {

String value() default "xxx";

}

這里是注解的一個簡單的例子,在接口前面加上一個@,就能定義一個注解了。

在這個注解中還有一個value()的成員變量,其中我們為它賦了默認值“xxx”,如果沒有默認值,那么該注解使用的時候,就必須為它傳值。

注解上面還有兩個注解,我們將之稱為元注解。

元注解

元注解,即在定義注解時,注解類也能夠使用其他的注解聲明。這種對注解類型進行注解的注解類,我們稱之為 meta-annotation(元注解)。

聲明的注解允許作用于哪些節點使用@Target聲明,例如ElementType.FIELD允許在成員變量上使用,而@Target注解是一個一對多的關系,即我們所寫的注解可以在多個地方定義,在類上定義,在方法上定義,等等;

保留級別由@Retention 聲明。其中保留級別如下。

RetentionPolicy.SOURCE

標記的注解僅保留在源級別中,并被編譯器忽略。

RetentionPolicy.CLASS

標記的注解在編譯時由編譯器保留,但 Java 虛擬機(JVM)會忽略。

RetentionPolicy.RUNTIME

標記的注解由 JVM 保留,因此運行時環境可以使用它。

當我們使用@SOURCE聲明注解時候,@Example注解的保留級別為SOURSE,即保留到源碼階段。

@Example("123")

public class MainActivity extends AppCompatActivity {

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

}

}

通過ASM反編譯工具查看MainActivity.class字節碼,并沒有看到注解的存在,因為在編譯過程中,注解已經被抹除了

// 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 ()V

......

protected onCreate(Landroid/os/Bundle;)V

......

}

注解的應用場景

根據注解的保留級別不同,對注解的使用自然存在不同場景。由注解的三個不同保留級別可知,注解作用于:

源碼、字節碼與運行時可以產生不同的應用場景。

級別

技術

說明

源碼

APT

在編譯期能夠獲取注解與注解聲明的類包括類中所有成員信息,一般用于生成額外的輔助類

字節碼

字節碼增強

在編譯出Class后,通過修改Class數據以實現修改代碼邏輯目的。對于是否需要修改的區分或者修改為不同邏輯的判斷可以使用注解

運行時

反射

在程序運行期間,通過反射技術動態獲取注解與其元素,從而完成不同的邏輯判定

APT技術

APT技術,APT,全稱為Annotation Processor Tools ,即注解處理器

要定義一個注解處理器,首先要新建一個普通Java模塊,并在app模塊中依賴。

在其中新建一個注解處理器,繼承自AbstractProcessor,編譯器已經為我們在內部實現了注解的采集,我們只需要對注解進行處理就可以了。

@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中注冊一樣,注解器也需要配置才可以生效。配置文件層級為compiler/main/resources/META-INF/services/javax.annotation.processing.Processor

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

com.example.compile.ExampleProcessor

注解處理程序運行在什時候

我們知道,一個.java文件要由javac編譯成.class文件,并交由虛擬機去運行。在這個過程中,javac會采集到所有的注解信息,并包裝成Element節點,然后交給注解處理程序。那么怎么證明這一點呢?

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

Android注解語法檢查

在Android中我們需要設計接口以供使用者調用時,如出現需要對入參進行類型限定,如限定為資源ID、布局ID等類型參數,將參數類型直接給定int即可。然而,我們可以利用Android為我們提供的語法檢查注解,來輔助進行更為直接的參數類型檢查與提示。

如參數限制為:圖片資源ID。這里利用了@Drawable 來限定入參為Drawable類型的int值

public Drawable getMyDrawable(@DrawableRes int id) {

return getDrawable(id);

}

在平時開發中假如有一個方法限制了入參的類型,那么我們可以使用枚舉來解決。

private Weekday currentDay;

enum Weekday {

SUNDAY,MONDAY

}

public void setCurrentDay(Weekday currentDay){

this.currentDay=currentDay;

}

但是通過ASM字節碼工具可以發現,枚舉其實是生成了對象,較int基本數據類型會比較占用內存

// 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

這時候就可以使用@IntDef注解來進行語法檢查,@IntDef是AndroidX為我們提供的一個元注解。由IDE來實現,在我們編寫代碼的時候進行檢查。

@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;

}

但是,語法檢查階段對于我們編譯是不會產生影響的。

字節碼增強技術

什么叫字節碼增強技術?就是在字節碼中寫代碼。平時我們是在.java文件中去編寫代碼,其有一定的格式,也正如.java一樣,.class文件也有一定的格式(數據按照特定的方式記錄與排列)。

QQ空間曾經發布的熱修復解決方案中利用Javaassist 庫實現向類的構造函數中插入一段代碼解決

CLASS_ISPREVERIFIED 問題。包括了Instant Run的實現以及參照Instant Run實現的熱修復美團Robus等等等等都利用到了插樁技術。

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

由于對于該技術也只是有所了解,因此不做更多贅述。如果大家感興趣可以自己去加強學習。

利用注解加反射實現findViewById

我們利用注解加反射來實現一個簡單的findViewById。

首先我們定義一個@InjectView的注解 ,里面包含一個@Idres int類型的成員變量,用來存放控件的id值。

由于程序需要在運行期間利用反射來獲取元素的注解和值,因此注解應聲明在Runtime階段執行

@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);

//反射設置屬性的值

declaredField.setAccessible(true);//設置訪問權限,允許操作private屬性

try {

declaredField.set(activity, view);

} catch (IllegalAccessException e) {

e.printStackTrace();

}

}

}

}

}

InjectViewUtil在處理時,先通過getClass拿到Activity的類對象,再使用getDeclaredFields拿到其成員變量,并通過if (declaredField.isAnnotationPresent(InjectView.class))

判斷是否被@InjectView注解過了,并進一步拿到id值,最后使用set方法設置回去。運行項目,觀察效果,TextView 對象已經獲取到了實例,并修改為了我們設置的text值。

這是ButterKnife早期的實現,但由于運行階段利用反射去處理注解,會影響運行時的性能,所以后面它是在編譯時對注解進行解析完成相關代碼的生成,即剛剛介紹的第一種,利用注解處理器去完成findViewById的過程,相關源碼大家感興趣的話可以去查閱。

關于java注解的知識本次就介紹到這~

じゃ、また

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/532844.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/532844.shtml
英文地址,請注明出處:http://en.pswp.cn/news/532844.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

C/C++中的typedef 和 #define

C/C中的typedef 和 #define typedef C/C中的關鍵字typedef允許用戶為類型名來起一個新名字,通常會是縮寫或者能夠清晰表明類型含義的新名字。 例: typedef unsigned int UINT; UINT 100;值得注意的是,typedef除了為C/C內置的數據類型取別…

php3.2.3 升級,thinkphp3.2.3 升級到3.2.4時出錯問題

有些項目最初用OneThink做的,而OneThink 默認使用的TP 是3.2.0 的,沒事的時候就想給升級一下,但是直接復制進去的時候,有錯誤,導致OneThink 不能運行,排查后,需要修改兩個地方1、修改 Applicati…

Positional Encodings in ViTs 近期各視覺Transformer中的位置編碼方法總結及代碼解析 1

Positional Encodings in ViTs 近期各視覺Transformer中的位置編碼方法總結及代碼解析 最近CV領域的Vision Transformer將在NLP領域的Transormer結果借鑒過來,屠殺了各大CV榜單。對其做各種改進的頂會論文也是層出不窮,本文將聚焦于各種最新的視覺trans…

mysql 分析查詢語句,MySQL教程之SQL語句分析查詢優化

怎么獲取有功能問題的SQL1、經過用戶反應獲取存在功能問題的SQL2、經過慢查詢日志獲取功能問題的SQL3、實時獲取存在功能問題的SQL運用慢查詢日志獲取有功能問題的SQL首要介紹下慢查詢相關的參數1、slow_query_log 發動定制記載慢查詢日志設置的辦法,能夠經過MySQL指…

關于PyTorch中的register_forward_hook()函數未能執行其中hook函數的問題

關于PyTorch中的register_forward_hook()函數未能執行其中hook函數的問題 Hook 是 PyTorch 中一個十分有用的特性。利用它,我們可以不必改變網絡輸入輸出的結構,方便地獲取、改變網絡中間層變量的值和梯度。這個功能被廣泛用于可視化神經網絡中間層的 f…

geoda權重矩陣導入matlab,空間計量經濟學-分析解析.ppt

廈門大學 鄧明 空間截面回歸模型 地理加權回歸模型 地理加權回歸模型擴展了普通線性回歸模型。在GWR模型中,特定區位的回歸系數不再是利用全部信息獲得的假定常數,而是利用鄰近觀測值的子樣本數據信息進行局域(Local)回歸估計而得,并隨著空間…

樹莓派攝像頭基礎配置及測試

樹莓派攝像頭基礎配置 step 1 硬件連接 硬件連接,注意不要接反了,排線藍色一段朝向網口的方向。(筆者的設備是樹莓派4B) step 2 安裝raspi-config 安裝 raspi-config raspi-config在raspbian中是預裝的,而在kali、…

matlab sobel銳化,sobel銳化 - yirui wu.ppt

sobel銳化 - yirui wu第六章 圖像銳化 圖像銳化的概念 圖像銳化的目的是加強圖像中景物的細節邊緣和輪廓。 銳化的作用是使灰度反差增強。 因為邊緣和輪廓都位于灰度突變的地方。所以銳化算法的實現是基于微分作用。 圖像銳化方法 圖像的景物細節特征; 一階微分銳化…

使用百度云智能SDK和樹莓派搭建簡易的人臉識別系統 Python語言版

硬件 樹莓派4B一個CSI攝像頭一個 筆者使用的是樹莓派4B和CSI攝像頭,但是樹莓派3和USB攝像頭等相似設備均可。 百度云智能設置 Step 1 登錄 百度云智能 網址https://cloud.baidu.com/ 首先登錄百度賬號,與百度云、百度貼吧等互通,可直接…

php 5.6 引用傳遞,升級到5.6.x后如何在php中修復引用傳遞

我最近將fom php 5.2升級到5.6,并且有一些代碼我無法修復://Finds users with the same ip- or email-addressfunction find_related_users($user_id) {global $pdo;//print_R($pdo);//Let SQL do the magic!$sth $pdo->prepare(CALL find_related_users(?));$…

RuntimeError: [enforce fail at inline_container.cc:145] . PytorchStreamReader failed reading zip arc

RuntimeError: [enforce fail at inline_container.cc:145] . PytorchStreamReader failed reading zip archive: failed finding central directory 原因分析 這個報錯是出現在PyTorch在讀入模型參數時: checkpoint torch.load(epoch_15.pth, map_locationcpu)…

xp搭建 php環境,windows xp 下 LAMP環境搭建

1. apache安裝步驟如下圖在瀏覽器中輸入:localhost,出現下面頁面說明已成功安裝apache。2. mysql安裝如下圖顯示在運行里面輸入cmd ,然后連接測試mysql ,如圖所示:3. php安裝(1)將php壓縮包解壓到安裝路徑中的php目錄…

C++中的虛函數(表)實現機制以及用C語言對其進行的模擬實現

C中的虛函數(表)實現機制以及用C語言對其進行的模擬實現 聲明:本文非博主原創,轉自https://blog.twofei.com/496/,博主讀后受益良多,特地轉載,一是希望好文能有更多人看到,二是為了日后自己查閱。 前言 …

php 前端模板 yii,php – Yii2高級模板:添加獨立網頁

我在backend / views / site下添加了help.php,并在SiteController.php下聲明了一個能夠識別鏈接的函數public function behaviors(){return [access > [class > AccessControl::className(),rules > [[actions > [login, error],allow > true,],[actions > […

C++中數組和指針的關系(區別)詳解

C中數組和指針的關系(區別)詳解 本文轉自:http://c.biancheng.net/view/1472.html 博主在閱讀后將文中幾個知識點提出來放在前面: 沒有方括號和下標的數組名稱實際上代表數組的起始地址,這意味著數組名稱實際上就是…

安裝php獨立環境,0507-php獨立環境的安裝與配置 Web程序 - 貪吃蛇學院-專業IT技術平臺...

1.在一個純英文目錄下新建三個文件夾2.安裝apache(選擇好版本)過程中該填的按格式填好,其余的只更改安裝目錄即可如果報錯1901是安裝版本的問題。檢查:安裝完成后localhost打開為It works!添加到電腦屬性環境變量:3.將php文件解壓文檔放到AMP…

linux中PATH變量-詳細介紹

轉自:https://blog.csdn.net/haozhepeng/article/details/100584451 轉載者勘誤 原文最后提到的 echo 命令對于環境變量的修改無影響。這是肯定的,echo 命令相當于只是一個打印的函數(比如 Python 中的 print)。這里要修改環境變…

php assert eval,代碼執行函數之一句話木馬

前言大家好,我是阿里斯,一名IT行業小白。非常抱歉,昨天的內容出現瑕疵比較多,今天重新整理后再次發出,修改并添加了細節,另增加了常見的命令執行函數如果哪里不足,還請各位表哥指出。eval和asse…

顯卡、顯卡驅動、CUDA、CUDA Toolkit、cuDNN 梳理

顯卡、顯卡驅動、CUDA、CUDA Toolkit、cuDNN 梳理 轉自:https://www.cnblogs.com/marsggbo/p/11838823.html#nvccnvidia-smi GPU型號含義 顯卡: 簡單理解這個就是我們前面說的GPU,尤其指NVIDIA公司生產的GPU系列,因為后面介紹的…

php中msubstr,PHP學習:thinkphp中字符截取函數msubstr()用法分析

《PHP學習:thinkphp中字符截取函數msubstr()用法分析》要點:本文介紹了PHP學習:thinkphp中字符截取函數msubstr()用法分析,希望對您有用。如果有疑問,可以聯系我們。本文實例講述了thinkphp中字符截取函數msubstr()用法…