轉:https://www.cnblogs.com/acm-bingzi/p/javaAnnotation.html
什么是注解?
Annontation是Java5開始引入的新特征,中文名稱叫注解。它提供了一種安全的類似注釋的機制,用來將任何的信息或元數據(metadata)與程序元素(類、方法、成員變量等)進行關聯。為程序的元素(類、方法、成員變量)加上更直觀更明了的說明,這些說明信息是與程序的業務邏輯無關,並且供指定的工具或框架使用。Annontation像一種修飾符一樣,應用於包、類型、構造方法、方法、成員變量、參數及本地變量的聲明語句中。
Java注解是附加在代碼中的一些元信息,用於一些工具在編譯、運行時進行解析和使用,起到說明、配置的功能。注解不會也不能影響代碼的實際邏輯,僅僅起到輔助性的作用。包含在 java.lang.annotation 包中。
注解的用處:
1、生成文檔。這是最常見的,也是java 最早提供的注解。常用的有@param @return 等
2、跟蹤代碼依賴性,實現替代配置文件功能。比如Dagger 2 依賴注入,未來java 開發,將大量注解配置,具有很大用處;
3、在編譯時進行格式檢查。如@override 放在方法前,如果你這個方法並不是覆蓋了超類方法,則編譯時就能檢查出。
注解的原理:
注解本質是一個繼承了Annotation 的特殊接口,其具體實現類是Java 運行時生成的動態代理類。而我們通過反射獲取注解時,返回的是Java 運行時生成的動態代理對象$Proxy1。通過代理對象調用自定義注解(接口)的方法,會最終調用AnnotationInvocationHandler 的invoke 方法。該方法會從memberValues 這個Map 中索引出對應的值。而memberValues 的來源是Java 常量池。
元注解:
java.lang.annotation 提供了四種元注解,專門注解其他的注解(在自定義注解的時候,需要使用到元注解):
@Documented – 注解是否將包含在JavaDoc中
@Retention – 什么時候使用該注解
@Target – 注解用於什么地方
@Inherited – 是否允許子類繼承該注解
1.)@Retention – 定義該注解的生命周期
● RetentionPolicy.SOURCE : 在編譯階段丟棄。這些注解在編譯結束之后就不再有任何意義,所以它們不會寫入字節碼。@Override, @SuppressWarnings都屬於這類注解。
● RetentionPolicy.CLASS : 在類加載的時候丟棄。在字節碼文件的處理中有用。注解默認使用這種方式
● RetentionPolicy.RUNTIME : 始終不會丟棄,運行期也保留該注解,因此可以使用反射機制讀取該注解的信息。我們自定義的注解通常使用這種方式。
2.)Target – 表示該注解用於什么地方。默認值為任何元素,表示該注解用於什么地方。可用的ElementType 參數包括
● ElementType.CONSTRUCTOR: 用於描述構造器
● ElementType.FIELD: 成員變量、對象、屬性(包括enum實例)
● ElementType.LOCAL_VARIABLE: 用於描述局部變量
● ElementType.METHOD: 用於描述方法
● ElementType.PACKAGE: 用於描述包
● ElementType.PARAMETER: 用於描述參數
● ElementType.TYPE: 用於描述類、接口(包括注解類型) 或enum聲明
3.)@Documented – 一個簡單的Annotations 標記注解,表示是否將注解信息添加在java 文檔中。
4.)@Inherited – 定義該注釋和子類的關系
@Inherited 元注解是一個標記注解,@Inherited 闡述了某個被標注的類型是被繼承的。如果一個使用了@Inherited 修飾的annotation 類型被用於一個class,則這個annotation 將被用於該class 的子類。
自定義注解:
自定義注解類編寫的一些規則:
1. Annotation 型定義為@interface, 所有的Annotation 會自動繼承java.lang.Annotation這一接口,並且不能再去繼承別的類或是接口.
2. 參數成員只能用public 或默認(default) 這兩個訪問權修飾
3. 參數成員只能用基本類型byte、short、char、int、long、float、double、boolean八種基本數據類型和String、Enum、Class、annotations等數據類型,以及這一些類型的數組.
4. 要獲取類方法和字段的注解信息,必須通過Java的反射技術來獲取 Annotation 對象,因為你除此之外沒有別的獲取注解對象的方法
5. 注解也可以沒有定義成員,,不過這樣注解就沒啥用了
PS:自定義注解需要使用到元注解
自定義注解demo:
水果名稱注解:
1 import java.lang.annotation.Documented; 2 import java.lang.annotation.Retention; 3 import java.lang.annotation.Target; 4 import static java.lang.annotation.ElementType.FIELD; 5 import static java.lang.annotation.RetentionPolicy.RUNTIME; 6 7 /** 8 * 水果名稱注解 9 */ 10 @Target(FIELD) 11 @Retention(RUNTIME) 12 @Documented 13 public @interface FruitName { 14 String value() default ""; 15 }
水果顏色注解:
1 import java.lang.annotation.Documented; 2 import java.lang.annotation.Retention; 3 import java.lang.annotation.Target; 4 import static java.lang.annotation.ElementType.FIELD; 5 import static java.lang.annotation.RetentionPolicy.RUNTIME; 6 7 /** 8 * 水果顏色注解 9 */ 10 @Target(FIELD) 11 @Retention(RUNTIME) 12 @Documented 13 public @interface FruitColor { 14 /** 15 * 顏色枚舉 16 */ 17 public enum Color{ BLUE,RED,GREEN}; 18 19 /** 20 * 顏色屬性 21 */ 22 Color fruitColor() default Color.GREEN; 23 24 }
注解元素的默認值:
注解元素必須有確定的值,要么在定義注解的默認值中指定,要么在使用注解時指定,非基本類型的注解元素的值不可為null。因此, 使用空字符串或0作為默認值是一種常用的做法。這個約束使得處理器很難表現一個元素的存在或缺失的狀態,因為每個注解的聲明中,所有元素都存在,並且都具有相應的值,為了繞開這個約束,我們只能定義一些特殊的值,例如空字符串或者負數,一次表示某個元素不存在,在定義注解時,這已經成為一個習慣用法。
水果供應商注解:
1 import java.lang.annotation.Documented; 2 import java.lang.annotation.Retention; 3 import java.lang.annotation.Target; 4 import static java.lang.annotation.ElementType.FIELD; 5 import static java.lang.annotation.RetentionPolicy.RUNTIME; 6 7 8 /** 9 * 水果供應者注解 10 */ 11 @Target(FIELD) 12 @Retention(RUNTIME) 13 @Documented 14 public @interface FruitProvider { 15 /** 16 * 供應商編號 17 */ 18 public int id() default -1; 19 20 /** 21 * 供應商名稱 22 */ 23 public String name() default ""; 24 25 /** 26 * 供應商地址 27 */ 28 public String address() default ""; 29 }
注解處理器:基於反射機制獲取注解信息
1 import java.lang.reflect.Field; 2 3 /** 4 * 注解處理器 5 */ 6 public class FruitInfoUtil { 7 public static void getFruitInfo(Class<?> clazz){ 8 9 String strFruitName=" 水果名稱:"; 10 String strFruitColor=" 水果顏色:"; 11 String strFruitProvicer="供應商信息:"; 12 13 Field[] fields = clazz.getDeclaredFields(); 14 15 for(Field field :fields){ 16 if(field.isAnnotationPresent(FruitName.class)){ 17 FruitName fruitName = (FruitName) field.getAnnotation(FruitName.class); 18 strFruitName=strFruitName+fruitName.value(); 19 System.out.println(strFruitName); 20 } 21 else if(field.isAnnotationPresent(FruitColor.class)){ 22 FruitColor fruitColor= (FruitColor) field.getAnnotation(FruitColor.class); 23 strFruitColor=strFruitColor+fruitColor.fruitColor().toString(); 24 System.out.println(strFruitColor); 25 } 26 else if(field.isAnnotationPresent(FruitProvider.class)){ 27 FruitProvider fruitProvider= (FruitProvider) field.getAnnotation(FruitProvider.class); 28 strFruitProvicer=" 供應商編號:"+fruitProvider.id()+" 供應商名稱:"+fruitProvider.name()+" 供應商地址:"+fruitProvider.address(); 29 System.out.println(strFruitProvicer); 30 } 31 } 32 } 33 }
注解使用:
1 import test.FruitColor.Color; 2 3 /** 4 * 注解使用 5 */ 6 public class Apple { 7 8 @FruitName("Apple") 9 private String appleName; 10 11 @FruitColor(fruitColor=Color.RED) 12 private String appleColor; 13 14 @FruitProvider(id=1,name="陝西紅富士集團",address="陝西省西安市延安路89號紅富士大廈") 15 private String appleProvider; 16 17 public void setAppleColor(String appleColor) { 18 this.appleColor = appleColor; 19 } 20 public String getAppleColor() { 21 return appleColor; 22 } 23 24 public void setAppleName(String appleName) { 25 this.appleName = appleName; 26 } 27 public String getAppleName() { 28 return appleName; 29 } 30 31 public void setAppleProvider(String appleProvider) { 32 this.appleProvider = appleProvider; 33 } 34 public String getAppleProvider() { 35 return appleProvider; 36 } 37 38 public void displayName(){ 39 System.out.println("水果的名字是:蘋果"); 40 } 41 }
1 /** 2 * 輸出結果 3 */ 4 public class FruitRun { 5 public static void main(String[] args) { 6 FruitInfoUtil.getFruitInfo(Apple.class); 7 } 8 }
運行結果:
水果名稱:Apple
水果顏色:RED
供應商編號:1 供應商名稱:陝西紅富士集團 供應商地址:陝西省西安市延安路89號紅富士大廈