zoukankan      html  css  js  c++  java
  • 深入理解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号红富士大厦
    
    本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利
  • 相关阅读:
    3、linux网络编程--网络字节序
    1、linux网络编程--网络协议
    第四篇:进程
    第五篇:进程通信
    第六篇:线程原理
    操作系统基本概念
    第一篇:基础原理篇
    第二篇:操作系统历史
    (2)linux下的简单的socket通信实例
    (3)基于linux的socket编程TCP半双工client-server聊天程序
  • 原文地址:https://www.cnblogs.com/hhddd-1024/p/15326918.html
Copyright © 2011-2022 走看看