深入理解Java:注解(Annotation)--注解处理器

  使用自定义注解的过程中,创建用来处理注解的注解处理器是非常重要的。使用反射机制的API,可以快速的构造自定义注解处理器。

注解处理器类库(java.lang.reflect.AnnotatedElement):

  Annotation接口代表各种程序元素前面的注解,该接口是所有Annotation类型的父接口。

  在java.lang.reflect 包下有一个AnnotatedElement接口,这个接口代表程序中可以接受注解的程序元素,该接口主要有如下几个实现类:

  Class:类定义

  Constructor:构造器定义

  Field:累的成员变量定义

  Method:类的方法定义

  Package:类的包定义

  java.lang.reflect 包下主要包含一些实现反射功能的工具类,并且这个包提供的所有反射的API都在一定程度上扩充了读取运行时Annotation信息的能力。当一个Annotation类型被定义为运行时的Annotation后,该注解才能是运行时可见,当class文件被装载时,保存在class文件中的Annotation才会被虚拟机读取。

  AnnotatedElement接口是所有程序元素(Class、Method和Constructor)的父接口,所以程序通过反射获取了某个类的AnnotatedElement对象之后,程序就可以调用该对象的如下四个个方法来访问Annotation信息:

  • 方法1: T getAnnotation(Class annotationClass): 返回当前程序元素上存在的、指定类型的注解,如果该类型注解不存在,则返回null。
  • 方法2:Annotation[] getAnnotations():返回该程序元素上存在的所有注解。
  • 方法3:boolean isAnnotationPresent(Class<?extends Annotation> annotationClass):判断该程序元素上是否包含指定类型的注解,存在则返回true,否则返回false。
  • 方法4:Annotation[] getDeclaredAnnotations():返回直接存在于此元素上的所有注释。与此接口中的其他方法不同,该方法将忽略继承的注释。(如果没有注释直接存在于此元素上,则返回长度为零的一个数组。)该方法的调用者可以随意修改返回的数组;这不会对其他调用者返回的数组产生任何影响。

以下是一个简单的注解处理器:

  • 注解声明
/**
 * 水果名称注解
 */
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface FruitName {
    String value() default "";
}
--------------------------------------------文件分割-----------------------------------------------------------
/**
 * 水果颜色注解
 */
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface FruitColor {
    /**
     * 颜色枚举
     */
    public enum Color{ BULE,RED,GREEN};
    
    /**
     * 颜色属性
     */
    Color fruitColor() default Color.GREEN;

}
--------------------------------------------文件分割-----------------------------------------------------------
/**
 * 水果供应者注解
 */
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface FruitProvider {
    /**
     * 供应商编号
     */
    public int id() default -1;
    
    /**
     * 供应商名称
     */
    public String name() default "";
    
    /**
     * 供应商地址
     */
    public String address() default "";
}
  • 注解使用
public class Apple {  // 苹果实体类
    
    @FruitName("Apple")
    private String appleName;
    
    @FruitColor(fruitColor=Color.RED)
    private String appleColor;
    
    @FruitProvider(id=1,name="陕西红富士集团",address="陕西省西安市延安路89号红富士大厦")
    private String appleProvider;
    
    public void setAppleColor(String appleColor) {
        this.appleColor = appleColor;
    }
    public String getAppleColor() {
        return appleColor;
    }
    
    public void setAppleName(String appleName) {
        this.appleName = appleName;
    }
    public String getAppleName() {
        return appleName;
    }
    
    public void setAppleProvider(String appleProvider) {
        this.appleProvider = appleProvider;
    }
    public String getAppleProvider() {
        return appleProvider;
    }
    
    public void displayName(){
        System.out.println("水果的名字是:苹果");
    }
}
  • 注解处理器
public class FruitInfoUtil {
    public static void getFruitInfo(Class<?> clazz){
        String strFruitName=" 水果名称:";
        String strFruitColor=" 水果颜色:";
        String strFruitProvicer="供应商信息:";
        
        // 通过类对象拿到所有字段对象
        Field[] fields = clazz.getDeclaredFields();
        
        // 遍历每一个字段对象
        for(Field field :fields){
            // 判断字段上是否存在FruitName.class注解
            if(field.isAnnotationPresent(FruitName.class)){
                // 拿到FruitName.class注解
                FruitName fruitName = (FruitName) field.getAnnotation(FruitName.class);
                // 拿到FruitName.class注解标记时value成员的值
                strFruitName=strFruitName+fruitName.value();
                System.out.println(strFruitName);
            }// 判断字段上是否存在FruitColor.class注解
            else if(field.isAnnotationPresent(FruitColor.class)){
                FruitColor fruitColor= (FruitColor) field.getAnnotation(FruitColor.class);
                strFruitColor=strFruitColor+fruitColor.fruitColor().toString();
                System.out.println(strFruitColor);
            } // 同上
            else if(field.isAnnotationPresent(FruitProvider.class)){
                FruitProvider fruitProvider= (FruitProvider) field.getAnnotation(FruitProvider.class);
                strFruitProvicer=" 供应商编号:"+fruitProvider.id()+" 供应商名称:"+fruitProvider.name()+" 供应商地址:"+fruitProvider.address();
                System.out.println(strFruitProvicer);
            }
        }
    }
}
  • 测试
public static void main(String[] args) {
	FruitInfoUtil.getFruitInfo(Apple.class);
}

===============控制台输出====================
 水果名称:Apple
 水果颜色:RED
 供应商编号:1 供应商名称:陕西红富士集团 供应商地址:陕西省西安市延安路89号红富士大厦
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利
原文地址:https://www.cnblogs.com/hhddd-1024/p/15326918.html