java Annotation 註解

來源:互聯網
上載者:User

首先什麼是註解?

  最常見的是,在我們使用Eclipse等工具編寫java代碼的時候,有時候會出現一些比如@Deprecated,@Override,@SuppressWarnings等東東。這個就是常見的幾種註解。


  在開發Java程式,尤其是Java EE應用的時候,總是免不了與各種設定檔打交道。以Java EE中典型的S(pring)S(truts)H(ibernate)架構來說,Spring、Struts和Hibernate這三個架構都有自己的XML格式的設定檔。這些設定檔需要與Java原始碼儲存同步,否則的話就可能出現錯誤。而且這些錯誤有可能到了運行時刻才被發現。把同一份資訊儲存在兩個地方,總是個壞的主意。理想的情況是在一個地方維護這些資訊就好了。其它部分所需的資訊則通過自動的方式來產生。JDK 5中引入了原始碼中的註解(annotation)這一機制。註解使得Java原始碼中不但可以包含功能性的實現代碼,還可以添加中繼資料。註解的功能類似於代碼中的注釋,所不同的是註解不是提供代碼功能的說明,而是實現程式功能的重要組成部分。Java註解已經在很多架構中得到了廣泛的使用,用來簡化程式中的配置。

  而我們最常見的可能就是上面提到的這三個註解了,簡單的介紹一下上面的這三個註解的作用:

    @Override:只能用在方法之上的,用來告訴別人這一個方法是改寫父類的。 
    @Deprecated:建議別人不要使用舊的API的時候用的,編譯的時候會用產生警告資訊,可以設定在程式裡的所有的元素上. 
        @SuppressWarnings:這一個類型可以來暫時把一些警告資訊訊息關閉. 

在jdk內建的java.lang.annotation包裡,開啟如下幾個源檔案:Target.java,Retention.java,RetentionPolicy.java,ElementType.java。

內容分別為:

@Documented  @Retention(RetentionPolicy.RUNTIME)   @Target(ElementType.ANNOTATION_TYPE)   public @interface Target {      ElementType[] value();   }  

  

@Documented  @Retention(RetentionPolicy.RUNTIME)   @Target(ElementType.ANNOTATION_TYPE)   public @interface Retention {      RetentionPolicy value();   }  

  

public enum RetentionPolicy {    SOURCE,    CLASS,    RUNTIME   }  

  

public enum ElementType {   TYPE, FIELD, METHOD, PARAMETER, CONSTRUCTOR,   LOCAL_VARIABLE, ANNOTATION_TYPE,PACKAGE  } 

  在設計annotations的時候必須把一個類型定義為@interface。

     我們需要注意的是:SOURCE,CLASS 和 RUNTIME.這三個層級。

     SOURCE代表的是這個Annotation類型的資訊只會保留在程式源碼裡,源碼如果經過了編譯之後,Annotation的資料就會消失,並不會保留在編譯好的.class檔案裡面。 
         ClASS的意思是這個Annotation類型的資訊保留在程式源碼裡,同時也會保留在編譯好的.class檔案裡面,在執行的時候,並不會把這一些資訊載入到虛擬機器(JVM)中去.注意一下,當你沒有設定一個Annotation類型的Retention值時,系統預設值是CLASS. 
         RUNTIME,表示在源碼、編譯好的.class檔案中保留資訊,在執行的時候會把這一些資訊載入到JVM中去的.

    @Target裡面的ElementType是用來指定Annotation類型可以用在哪一些元素上的.說明一下:TYPE(類型), FIELD(屬性), METHOD(方法), PARAMETER(參數), CONSTRUCTOR(建構函式),LOCAL_VARIABLE(局部變數), ANNOTATION_TYPE,PACKAGE(包),其中的TYPE(類型)是指可以用在Class,Interface,Enum和Annotation類型上.
     另外,@Target自己也用了自己來聲明自己。如果一個Annotation類型沒有指明@Target使用在哪些元素上,那麼它可以使用在任何元素之上,這裡的元素指的是上面的八種類型. 
   舉幾個正確的例子:

@Target(ElementType.METHOD)    @Target(value=ElementType.METHOD)    @Target(ElementType.METHOD,ElementType.CONSTRUCTOR)

  @Documented的目的就是讓這一個Annotation類型的資訊能夠顯示在javaAPI說明文檔上;沒有添加的話,使用javadoc產生API文檔的時候就會找不到這一個類型產生的資訊.

  另外一點,如果需要把Annotation的資料繼承給子類,那麼就會用到@Inherited這一個Annotation類型.

     本文只是簡單的說了一下註解的常規用法,至於更加深入的註解學習,請參見文章末尾的參考資料。下面我們來看自訂一個註解:原始碼有如下幾個:

源碼分別為:

package com.java.annotation;import java.lang.annotation.ElementType;import java.lang.annotation.Retention;import java.lang.annotation.RetentionPolicy;import java.lang.annotation.Target;/** * 類註解 * */@Retention(RetentionPolicy.RUNTIME)@Target(ElementType.TYPE)public @interface MyAnnotationClass {public String msg();}

  

package com.java.annotation;import java.lang.annotation.ElementType;import java.lang.annotation.Retention;import java.lang.annotation.RetentionPolicy;import java.lang.annotation.Target;/** * 方法註解 * */@Retention(RetentionPolicy.RUNTIME)@Target(ElementType.METHOD)public @interface MyAnnotationMethod {public String common();}

  

package com.java.annotation;import java.lang.annotation.ElementType;import java.lang.annotation.Retention;import java.lang.annotation.RetentionPolicy;import java.lang.annotation.Target;@Retention(RetentionPolicy.RUNTIME)@Target(ElementType.FIELD)public @interface MyAnnotationField {boolean request();}

  

package com.java.annotation;@MyAnnotationClass(msg = "這個是一個類註解")public class MyAnnotationDemo {public MyAnnotationDemo() {}public MyAnnotationDemo(String hello) {this.hello = hello;}@MyAnnotationMethod(common = "這個是一個方法註解")public void method() {System.out.println(hello);}@MyAnnotationField(request = true)private String hello;}

  

package com.java.annotation;import java.lang.reflect.Field;import java.lang.reflect.Method;public class MyAnnotationTest {public static void main(String[] args) {MyAnnotationDemo demo = new MyAnnotationDemo("hello rollen");MyAnnotationClass annotationClass = demo.getClass().getAnnotation(MyAnnotationClass.class);System.out.println(annotationClass.msg());Method method = null;try {method = demo.getClass().getMethod("method",new Class[0]);} catch (SecurityException e) {e.printStackTrace();} catch (NoSuchMethodException e) {e.printStackTrace();}MyAnnotationMethod annotationMethod = method.getAnnotation(MyAnnotationMethod.class);System.out.println(annotationMethod.common());Field field = null;try {field = demo.getClass().getDeclaredField("hello");} catch (SecurityException e) {e.printStackTrace();} catch (NoSuchFieldException e) {e.printStackTrace();}MyAnnotationField annotationField = field.getAnnotation(MyAnnotationField.class);System.out.println(annotationField.request());}}

  

參考資料:http://www.infoq.com/cn/articles/cf-java-annotation

              http://tiantian0521.blog.163.com/blog/static/4172088320118243436208/

    

 

 

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.