zoukankan      html  css  js  c++  java
  • Java的自定义注解及通过反射获取注解

    一、注解基本知识

      1、元注解:@Retention @Target @Document @Inherited

      2、Annotation型定义为@interface, 所有的Annotation会自动继承java.lang.Annotation这一接口,并且不能再去继承别的类或是接口。

      3、参数成员只能用public或默认(default)这两个访问权修饰

      4、参数成员只能用基本类型byte,short,char,int,long,float,double,boolean八种基本数据类型和StringEnumClassannotations等数据类型,以及这一些类型的数组。

      5、要获取类、方法和字段的注解信息,必须通过Java的反射技术来获取 Annotation对象,除此之外没有别的获取注解对象的方法

      6、注解也可以没有定义成员, 不过这样注解就没啥用了,只起到标识作用

    自定义注解类时, 可以指定目标 (类、方法、字段, 构造函数等) , 注解的生命周期(运行时,class文件或者源码中有效), 是否将注解包含在javadoc中及是否允许子类继承父类中的注解, 具体如下:

      1、@Target 表示该注解目标,可能的 ElemenetType 参数包括:

        ElemenetType.CONSTRUCTOR 构造器声明
        ElemenetType.FIELD 域声明(包括 enum 实例)
        ElemenetType.LOCAL_VARIABLE 局部变量声明
        ElemenetType.METHOD 方法声明
        ElemenetType.PACKAGE 包声明
        ElemenetType.PARAMETER 参数声明
        ElemenetType.TYPE 类,接口(包括注解类型)或enum声明

      2、@Retention 表示该注解的生命周期,可选的 RetentionPolicy 参数包括

        RetentionPolicy.SOURCE 注解将被编译器丢弃
        RetentionPolicy.CLASS 注解在class文件中可用,但会被JVM丢弃
        RetentionPolicy.RUNTIME JVM将在运行期也保留注释,因此可以通过反射机制读取注解的信息

      3、@Documented 指示将此注解包含在 javadoc 中

      4、@Inherited 指示允许子类继承父类中的注解

    二、在java中的使用

      2.1、定义注解

    package annotation;
    
    import java.lang.annotation.ElementType;
    import java.lang.annotation.Retention;
    import java.lang.annotation.RetentionPolicy;
    import java.lang.annotation.Target;
    
    public class MyAnnotation {
        /**
         * 注解类
         * 
         * @author suguoliang
         *
         */
        @Target(ElementType.TYPE)
        @Retention(RetentionPolicy.RUNTIME)
        public @interface MyClassAnnotation {
            String uri();
    
            String desc();
        }
    
        /**
         * 构造方法注解
         * 
         * @author suguoliang
         *
         */
        @Target(ElementType.CONSTRUCTOR)
        @Retention(RetentionPolicy.RUNTIME)
        public @interface MyConstructorAnnotation {
            String uri();
    
            String desc();
        }
    
        /**
         * 方法注解
         * 
         * @author suguoliang
         *
         */
        @Target(ElementType.METHOD)
        @Retention(RetentionPolicy.RUNTIME)
        public @interface MyMethodAnnotation {
            String uri();
    
            String desc();
        }
    
        /**
         * 字段注解
         * 
         * @author suguoliang
         *
         */
        @Target(ElementType.FIELD)
        @Retention(RetentionPolicy.RUNTIME)
        public @interface MyFieldAnnotation {
            String uri();
    
            String desc();
        }
    
        /**
         * 可以同时应用到类和方法上
         * 
         * @author 尐蘇
         *
         */
        @Target({ ElementType.TYPE, ElementType.METHOD })
        @Retention(RetentionPolicy.RUNTIME)
        public @interface MyClassAndMethodAnnotation {
            // 定义枚举
            public enum EnumType {
                util, entity, service, model
            }
    
            // 设置默认值
            public EnumType classType() default EnumType.util;
    
            // 数组
            int[] arr() default { 3, 7, 5 };
    
            String color() default "blue";
        }
    }

      2.2基本测试

    package annotation;
    
    import java.lang.reflect.Constructor;
    import java.lang.reflect.Field;
    import java.lang.reflect.Method;
    
    import annotation.MyAnnotation.MyClassAndMethodAnnotation;
    import annotation.MyAnnotation.MyClassAndMethodAnnotation.EnumType;
    import annotation.MyAnnotation.MyClassAnnotation;
    import annotation.MyAnnotation.MyConstructorAnnotation;
    import annotation.MyAnnotation.MyFieldAnnotation;
    import annotation.MyAnnotation.MyMethodAnnotation;
    
    @MyClassAnnotation(desc = "The Class", uri = "com.sgl.annotation")
    @MyClassAndMethodAnnotation(classType = EnumType.util)
    public class TestAnnotation {
        @MyFieldAnnotation(desc = "The Class Field", uri = "com.sgl.annotation#id")
        private String id;
    
        @MyConstructorAnnotation(desc = "The Class Constructor", uri = "com.sgl.annotation#constructor")
        public TestAnnotation() {
        }
    
        public String getId() {
            return id;
        }
    
        @MyMethodAnnotation(desc = "The Class Method", uri = "com.sgl.annotation#setId")
        public void setId(String id) {
            this.id = id;
        }
    
        @MyMethodAnnotation(desc = "The Class Method sayHello", uri = "com.sgl.annotation#sayHello")
        public void sayHello(String name) {
            if (name == null || name.equals("")) {
                System.out.println("hello world!");
            } else {
                System.out.println(name + "	:say hello world");
            }
        }
    
        public static void main(String[] args) throws Exception {
            Class<TestAnnotation> clazz = TestAnnotation.class;
            // 获取类注解
            MyClassAnnotation myClassAnnotation = clazz.getAnnotation(MyClassAnnotation.class);
            System.out.println(myClassAnnotation.desc() + "+" + myClassAnnotation.uri());
    
            // 获得构造方法注解
            Constructor<TestAnnotation> constructors = clazz.getConstructor(new Class[] {});// 先获得构造方法对象
            MyConstructorAnnotation myConstructorAnnotation = constructors.getAnnotation(MyConstructorAnnotation.class);// 拿到构造方法上面的注解实例
            System.out.println(myConstructorAnnotation.desc() + "+" + myConstructorAnnotation.uri());
    
            // 获得方法注解
            Method method = clazz.getMethod("setId", new Class[] { String.class });// 获得方法对象
            MyMethodAnnotation myMethodAnnotation = method.getAnnotation(MyMethodAnnotation.class);
            System.out.println(myMethodAnnotation.desc() + "+" + myMethodAnnotation.uri());
    
            // 获得字段注解
            Field field = clazz.getDeclaredField("id");// 暴力获取private修饰的成员变量
            MyFieldAnnotation myFieldAnnotation = field.getAnnotation(MyFieldAnnotation.class);
            System.out.println(myFieldAnnotation.desc() + "+" + myFieldAnnotation.uri());
        }
    }

      2.3通过反射解析

    package annotation;
    
    import java.lang.reflect.Method;
    import java.util.Arrays;
    
    import annotation.MyAnnotation.MyClassAndMethodAnnotation;
    import annotation.MyAnnotation.MyClassAndMethodAnnotation.EnumType;
    import annotation.MyAnnotation.MyClassAnnotation;
    import annotation.MyAnnotation.MyMethodAnnotation;
    
    public class ParseAnnotation {
        /**
         * 解析方法注解
         * 
         * @param clazz
         */
        public static <T> void parseMethod(Class<T> clazz) {
            try {
                T obj = clazz.newInstance();
                for (Method method : clazz.getDeclaredMethods()) {
                    MyMethodAnnotation methodAnnotation = method.getAnnotation(MyMethodAnnotation.class);
                    if (methodAnnotation != null) {
                        // 通过反射调用带有此注解的方法
                        method.invoke(obj, methodAnnotation.uri());
                    }
                    MyClassAndMethodAnnotation myClassAndMethodAnnotation = method
                            .getAnnotation(MyClassAndMethodAnnotation.class);
                    if (myClassAndMethodAnnotation != null) {
                        if (EnumType.util.equals(myClassAndMethodAnnotation.classType())) {
                            System.out.println("this is a util method");
                        } else {
                            System.out.println("this is a other method");
                        }
                        System.out.println(Arrays.toString(myClassAndMethodAnnotation.arr()));// 打印数组
                        System.out.println(myClassAndMethodAnnotation.color());// 输出颜色
                    }
                    System.out.println("		-----------------------");
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    
        public static <T> void parseType(Class<T> clazz) {
            try {
                MyClassAndMethodAnnotation myClassAndMethodAnnotation = clazz
                        .getAnnotation(MyClassAndMethodAnnotation.class);
                if (myClassAndMethodAnnotation != null) {
                    if (EnumType.util.equals(myClassAndMethodAnnotation.classType())) {
                        System.out.println("this is a util class");
                    } else {
                        System.out.println("this is a other class");
                    }
                }
                MyClassAnnotation myClassAnnotation = clazz.getAnnotation(MyClassAnnotation.class);
                if (myClassAnnotation != null) {
                    System.err.println(" class info: " + myClassAnnotation.uri());
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
    
        }
    
        public static void main(String[] args) {
            parseMethod(TestAnnotation.class);
            parseType(TestAnnotation.class);
        }
    }
  • 相关阅读:
    X264参考手册
    X264码率控制总结
    x264_param_t结构体解释,设置及对应函数位置
    FFmpeg与libx264 x264接口对应关系源代码分析
    x264源代码学习1:概述与架构分析
    X264使用指南
    X264学习1:简介
    PHPMailer 使用 中文乱码
    laravel学习之路5缓存
    laravel学习之路4artisan
  • 原文地址:https://www.cnblogs.com/a591378955/p/8350499.html
Copyright © 2011-2022 走看看