我們在前2篇文章中:
分別介紹了annotation的基本概念,以及怎麼樣自訂annotation。
本文介紹怎麼在運行期(runtime)動態解析annotation。
上面我們介紹過,annotation只是附加在代碼裡的附加資訊,annotation本身不會對代碼的執行產生任何影響。
這樣說來,annotation到底能起什麼作用呢?
1, 編譯工具或其他工具可以根據被附加在代碼裡的annotation資訊自動組建組態檔案或文檔等外部檔案。
比如,sun公司就提供了apt(Annotation Processing Tool)工具,apt工具是一個可以處理annotation的命令列工具,apt提供了在編譯期針對原始碼層級的解析,並可以在解析時產生新的原始碼和其他檔案,同時還可以對產生的原始碼進行編譯。
2, 其他程式可以在運行時動態解析將要被執行的程式裡的annotation資訊,並根據被附加的annotation資訊來執行不同的操作。
比如,EJB3規範就比較廣泛地使用了annotation特性。比如只要在POJO為class註明@Stateless注釋,EJB容器便會根據此annotation把該POJO註冊為無狀態的Session Bean。EJB3使用了annotation大大地簡化了EJB的開發和配置過程。我們會在其他文章裡專門介紹EJB Annotation的原理與使用方法,這裡不做詳述。
本文通過一個簡單地例子來說明怎麼在運行期動態解析annotation。Apt工具的使用我們會在近期其他文章裡對其加以介紹。
比如,我們定義了MyAnnotation3注釋:
MyAnnotation3.java
Java代碼
- package com.test.annotation;
- import java.lang.annotation.Annotation;
- import java.lang.annotation.Inherited;
- import java.lang.annotation.Retention;
- import java.lang.annotation.RetentionPolicy;
- @Retention(RetentionPolicy.RUNTIME)
- public @interface MyAnnotation3 {
- public String value();
- public String[] multiValues();
- int number() default 0;
- }
上面定義了一個名為MyAnnotation3的注釋。
我們再定義一個GetMyAnnotation類,該類使用了MyAnnotation3注釋:
GetMyAnnotation.java:
Java代碼
- package com.test.annotation.test;
-
- import java.lang.annotation.Annotation;
- import java.lang.reflect.Field;
- import java.lang.reflect.Method;
- import javax.ejb.EJB;
- import javax.naming.InitialContext;
- import javax.naming.NamingException;
- import com.test.annotation.MyAnnotation3;
-
- // 為GetMyAnnotation類附加MyAnnotation3 注釋
- @MyAnnotation3(value = "Class GetMyAnnotation", multiValues = { "1", "2" })
- public class GetMyAnnotation {
-
- // 為testField1屬性附加MyAnnotation3 注釋
- @MyAnnotation3(value = "call testField1", multiValues = { "1" }, number = 1)
- private String testField1;
-
- // 為testMethod1方法附加MyAnnotation3 注釋
- @MyAnnotation3(value = "call testMethod1", multiValues = { "1", "2" }, number = 1)
- public void testMethod1() {
- }
-
- @Deprecated
- @MyAnnotation3(value = "call testMethod2", multiValues = { "3", "4", "5" })
- public void testMethod2() {
- }
- }
上面的例子GetMyAnnotation非常簡單,裡面沒有任何功能,但是分別為類(class),屬性(field),方法(method)申明(附加)了MyAnnotation3 注釋。
下面我們用程式TestMyAnnotation3對GetMyAnnotation裡MyAnnotation3注釋進行解析。
運行時解析annotation
TestMyAnnotation3.java
Java代碼
- public class TestMyAnnotation3 {
-
- public static void main(String[] args) {
-
- System.out.println("--Class Annotations--");
-
- if (GetMyAnnotation.class.isAnnotationPresent(MyAnnotation3.class)) {
- System.out.println("[GetMyAnnotation].annotation:");
-
- MyAnnotation3 classAnnotation = GetMyAnnotation.class
- .getAnnotation(MyAnnotation3.class);
- printMyAnnotation3(classAnnotation);
- }
- System.out.println("--Fields Annotations--");
-
- Field[] fields = GetMyAnnotation.class.getDeclaredFields();
-
- for (Field field : fields) {
-
- if (field.isAnnotationPresent(MyAnnotation3.class)) {
-
- System.out.println("[GetMyAnnotation." + field.getName()
- + "].annotation:");
-
- MyAnnotation3 fieldAnnotation = field
- .getAnnotation(MyAnnotation3.class);
-
- printMyAnnotation3(fieldAnnotation);
- }
- }
- System.out.println("--Methods Annotations--");
-
- Method[] methods = GetMyAnnotation.class.getDeclaredMethods();
-
- for (Method method : methods) {
-
- System.out.println("[GetMyAnnotation." + method.getName()
- + "].annotation:");
-
- if (method.isAnnotationPresent(MyAnnotation3.class)) {
-
- MyAnnotation3 methodAnnotation = method
- .getAnnotation(MyAnnotation3.class);
-
- printMyAnnotation3(methodAnnotation);
- }
- }
- }
-
- private static void printMyAnnotation3(MyAnnotation3 annotation3) {
-
- if (annotation3 == null) {
- return;
- }
- System.out.println("{value=" + annotation3.value());
- String multiValues = "";
-
- for (String value : annotation3.multiValues()) {
- multiValues += "," + value;
- }
-
- System.out.println("multiValues=" + multiValues);
-
- System.out.println("number=" + annotation3.number() + "}");
- }
- }
輸出:
--Class Annotations--
[GetMyAnnotation].annotation:
{value=Class GetMyAnnotation
multiValues=,1,2
number=0}
--Fields Annotations--
[GetMyAnnotation.testField1].annotation:
{value=call testField1
multiValues=,1
number=1}
--Methods Annotations--
[GetMyAnnotation.testMethod1].annotation:
{value=call testMethod1
multiValues=,1,2
number=1}
[GetMyAnnotation.testMethod2].annotation:
{value=call testMethod2
multiValues=,3,4,5
number=0}
JDK1.5以後的版本提供的跟annotation有關的介面:
Java代碼
- interface java.lang.reflect.AnnotatedElement {
-
- boolean isAnnotationPresent(Class<? extends Annotation> annotationClass);
-
- <T extends Annotation> T getAnnotation(Class<T> annotationClass);
-
- Annotation[] getAnnotations();
-
- Annotation[] getDeclaredAnnotations();
-
- }
該介面主要用來取得附加在類(class),構造方法(constructor),屬性(field),方法(method),包(package)上的annotation資訊。
JDK1.5與此有關的幾個類都實現了AnnotatedElement介面:
所以可以利用反射(reflection)功能在程式裡動態解析附加的annotation資訊。
總結:
本文通過舉例簡單地說明了怎麼動態解析annotation,大家可以舉一反三,利用Java的annotation特性,完成更複雜功能等。