本文介绍怎么在运行期(runtime)动态解析annotation。
我们在前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
package com.test.annotation;
上面定义了一个名为MyAnnotation3的注释。
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; } 我们再定义一个GetMyAnnotation类,该类使用了MyAnnotation3注释: GetMyAnnotation.java:
package com.test.annotation.test;
上面的例子GetMyAnnotation非常简单,里面没有任何功能,但是分别为类(class),属性(field),方法(method)申明(附加)了MyAnnotation3 注释。
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() { } } 下面我们用程序TestMyAnnotation3对GetMyAnnotation里MyAnnotation3注释进行解析。 运行时解析annotation TestMyAnnotation3.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有关的接口:
interface java.lang.reflect.AnnotatedElement {
该接口主要用来取得附加在类(class),构造方法(constructor),属性(field),方法(method),包(package)上的annotation信息。
boolean isAnnotationPresent(Class<? extends Annotation> annotationClass); <T extends Annotation> T getAnnotation(Class<T> annotationClass); Annotation[] getAnnotations(); Annotation[] getDeclaredAnnotations(); } JDK1.5与此有关的几个类都实现了AnnotatedElement接口:
java.lang.reflect.AccessibleObject,
所以可以利用反射(reflection)功能在程序里动态解析附加的annotation信息。
java.lang.reflect.Class, java.lang.reflect.Constructor, java.lang.reflect.Field, java.lang.reflect.Method, java.lang.reflect.Package 总结: 本文通过举例简单地说明了怎么动态解析annotation,大家可以举一反三,利用Java的annotation特性,完成更复杂功能等。
以上文章转自:http://www.lifevv.com/
|