自定义注解与切面

自定义注解

@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Encryption {
    String value() default "";//注解类型元素
}

@Target注解:是限制此注解能够被运用到哪些代码上,比如方法,接口,属性…看下图

public enum ElementType {
    /** 类,接口(包括注解类型)或枚举的声明 */
    TYPE,
    /** 属性 */
    FIELD,
    /** 方法 */
    METHOD,
    /** 方法形式参数 */
    PARAMETER,
    /** 构造 */
    CONSTRUCTOR,
    /** 局部变量 */
    LOCAL_VARIABLE,
    /** 注解类型 */
    ANNOTATION_TYPE,
    /** 包 */
    PACKAGE,
    /**
     * 类型参数声明
     *
     * @since 1.8
     */
    TYPE_PARAMETER,
    /**
     * Use of a type
     *
     * @since 1.8
     */
    TYPE_USE
}

@Retention:指定注解的生命周期,看下图

public enum RetentionPolicy {
    /**
     * 编译器将丢弃注释。
     */
    SOURCE,
    /**
     * 注释将由编译器记录在类文件中
     * 但不需要在运行时由VM保留。这是默认设置
     * behavior.
     */
    CLASS,
    /**
     * 注释将由编译器记录在类文件中,并且
*在运行时由VM保留,因此可以反射地读取它们.
     *
     * @see java.lang.reflect.AnnotatedElement
     */
    RUNTIME
}

一般自定义注解用的都是RUNTIME
Documented:

指示类型为的注释将由javadoc记录
*以及默认情况下类似的工具。此类型应用于注释
*类型的声明,其注释会影响带注释的
*他们的客户的元素。如果类型声明用
*文档化后,其注释成为公共API的一部分
*注释元素的名称。

注解类型元素:用value作为名字的话在使用注解的时候可以直接使用:@注解名(注解值),其等效于:@注解名(value = 注解值)

public class UserDemo {
    @Encryption("123")
    public String seqNo;
}

因为用的是FIELD,也可以不加()

public class UserDemo {
    @Encryption
    public String seqNo;
}

接下来配置解析注解,通过poi进行操作

public class EncryptionAction {
    public static void main(String[] args) {
        try {
            //通过Class.forName进行实例化
            Class<?> aClass = Class.forName("com.example.demo");
            //通过反射getMethod用于获取
            Field stuMethod = aClass.getDeclaredField("seqNo");
            /**
             *  获取类注解
             *  Encryption myClassAnnotation = aClass.getAnnotation(Encryption.class);
             *  获得方法注解,如果参数为基本类型数据需要转换为相应的包装类型的对象,比如Integer要转为int.class
             *  Method method = clazz.getMethod("seqNoList", String.class );
             */
//            Method stuMethod = aClass.getMethod("seqNo",String.class);
            //判断是否有这个注解
            boolean annotationPresent = stuMethod.isAnnotationPresent(Encryption.class);
            if (annotationPresent){
                //通过此方法该注解的注解类型元素方法就可以获得配置时的值数据;
                Encryption encryption = stuMethod.getAnnotation(Encryption.class);
                System.out.println("value:"+encryption.value());
            }
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }catch (NoSuchFieldException e) {
            e.printStackTrace();
        }
    }
}

因为我用的是需要字段加密的属于多个字段,一个方法里面有多个注解可以通过getdeclaredannotations方法获取所有的注解进行判断后统一加密

 if ("获取的所有注解" instanceof "指定注解"){ }//可以用这个代替isAnnotationPresent
上一篇:SQLite Encryption(加密)新姿势


下一篇:[PHP] 解决 laravel "No application encryption key has been specified"