Java自定义注解

一直对自定义注解感到好奇,今天就觉得学一下,从网上搜了个文章(https://www.cnblogs.com/liangweiping/p/3837332.html),跟着例子跑了一遍,还挺有趣的,遂分享出来。以下示例,在此文章基础上改进而来,仅供参考。

一、注解的基础

1.注解的定义:Java文件叫做Annotation,用@interface表示。

2.元注解:@interface上面按需要注解上一些东西,包括@Retention、@Target、@Document、@Inherited四种。

3.注解的保留策略:

  @Retention(RetentionPolicy.SOURCE)   // 注解仅存在于源码中,在class字节码文件中不包含

  @Retention(RetentionPolicy.CLASS)     // 默认的保留策略,注解会在class字节码文件中存在,但运行时无法获得

  @Retention(RetentionPolicy.RUNTIME)  // 注解会在class字节码文件中存在,在运行时可以通过反射获取到

4.注解的作用目标:

  @Target(ElementType.TYPE)                      // 接口、类、枚举、注解

  @Target(ElementType.FIELD)                     // 字段、枚举的常量

  @Target(ElementType.METHOD)                 // 方法

  @Target(ElementType.PARAMETER)            // 方法参数

  @Target(ElementType.CONSTRUCTOR)       // 构造函数

  @Target(ElementType.LOCAL_VARIABLE)   // 局部变量

  @Target(ElementType.ANNOTATION_TYPE) // 注解

  @Target(ElementType.PACKAGE)               // 包

5.注解包含在javadoc中:

  @Documented

6.注解可以被继承:

  @Inherited

7.注解解析器:用来解析自定义注解。

 

二、通过注解进行赋值 - 注解在方法上(结合了工厂方法模式)

  1.自定义注解

package init;

import java.lang.annotation.*;

@Documented
@Inherited
@Target({ElementType.FIELD, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface Init {
    public String value() default "";
}

 

  2.在数据模型使用注解

package init;

public class User {
    private String name;
    private String age;

    public String getName() {
        return name;
    }

    @Init(value = "liang")
    public void setName(String name) {
        this.name = name;
    }

    public String getAge() {
        return age;
    }

    @Init(value = "23")
    public void setAge(String age) {
        this.age = age;
    }
}

  3.用“构造工厂”充当“注解解析器”

package init;

import java.lang.reflect.Method;

public class UserFactory {
    public static User create() {
        User user = new User();

        // 获取User类中所有的方法(getDeclaredMethods也行)
        Method[] methods = User.class.getDeclaredMethods();

        try {
            for (Method method : methods) {
                // 如果此方法有注解,就把注解里面的数据赋值到user对象
                if (method.isAnnotationPresent(Init.class)) {
                    Init init = method.getAnnotation(Init.class);
                    method.invoke(user, init.value());
                }
            }
        } catch (Exception e) {
            System.err.println("发生异常:" + e.getMessage());
            return null;
        }

        return user;
    }
}

  4.运行的代码

package init;

public class Test {
    public static void main(String[] args) {
//        testField();
        testMethod();
    }

    public static void testMethod(){
        User user = UserFactory.create();

        System.out.println(user.getName());
        System.out.println(user.getAge());
    }

}

  5.运行结果

liang
23

三、通过注解进行赋值 - 注解在属性上(结合了工厂方法模式)

  1.自定义注解

package init;

import java.lang.annotation.*;

@Documented
@Inherited
@Target({ElementType.FIELD, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface Init {
    public String value() default "";
}

 

  2.在数据模型使用注解

package init;

public class UserField {

    @Init(value = "liang")
    private String name;

    @Init(value = "23")
    private String age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getAge() {
        return age;
    }

    public void setAge(String age) {
        this.age = age;
    }
}

  3.用“构造工厂”充当“注解解析器”

package init;


import java.lang.reflect.Field;
import java.lang.reflect.Method;

public class UserFieldFactory {
    public static UserField create() {
        UserField userField = new UserField();

        // 获取User类中所有的方法(getDeclaredMethods也行)
        Field[] fields = UserField.class.getDeclaredFields();

        try {
            for (Field field : fields) {
                // 如果此方法有注解,就把注解里面的数据赋值到user对象
                if (field.isAnnotationPresent(Init.class)) {
                    Init init = field.getAnnotation(Init.class);

                    //设置字段的值
                    field.setAccessible(true);
                    field.set(userField, init.value());

                    //通过method设置字段的值
//                    String fieldName = field.getName();
//                    fieldName = fieldName.replaceFirst(fieldName.substring(0, 1), fieldName.substring(0, 1).toUpperCase());
//                    Method method = UserField.class.getMethod("set" + fieldName, String.class);
//                    method.invoke(userField, init.value());
                }
            }
        } catch (Exception e) {
            System.err.println("发生异常:" + e.getMessage());
            return null;
        }

        return userField;
    }
}

  4.运行的代码

package init;

public class Test {
    public static void main(String[] args) {
        testField();
//        testMethod();
    }

    public static void testField(){
        UserField user = UserFieldFactory.create();

        System.out.println(user.getName());
        System.out.println(user.getAge());
    }

}

  5.运行结果

liang
23

四、通过注解进行校验

  1.自定义注解

package valid;

import java.lang.annotation.*;

@Documented
@Inherited
@Target({ ElementType.FIELD, ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
public @interface Validate
{
    public int min() default 1;

    public int max() default 10;

    public boolean isNotNull() default true;

    public String minErrorMsg() default "";

    public String maxErrorMsg() default "";

    public String isNotNullErrorMsg() default "";
}

  2.在数据模型使用注解

package valid;

public class User {
    @Validate(min = 2, minErrorMsg = "名字最小长度校验不通过!", max = 5, maxErrorMsg = "名字最大长度校验不通过!", isNotNullErrorMsg = "名称不能为空")
    private String name;

    @Validate(isNotNull = false)
    private String age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getAge() {
        return age;
    }

    public void setAge(String age) {
        this.age = age;
    }
}

  3.注解解析器

package valid;

import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

public class UserCheck {
    public static boolean check(User user) {
        if (user == null) {
            System.out.println("!!校验对象为空!!");
            return false;
        }

        // 获取User类的所有属性(如果使用getFields,就无法获取到private的属性)
        Field[] fields = user.getClass().getDeclaredFields();

        try {
            for (Field field : fields) {
                // 如果属性有注解,就进行校验
                if (field.isAnnotationPresent(Validate.class)) {
                    Validate validate = field.getAnnotation(Validate.class);

                    //获取字段的值
                    field.setAccessible(true);
                    String fieldValue = (String) field.get(user);

                    if(fieldValue == null){
                        if (validate.isNotNull()) {
                            System.out.println(validate.isNotNullErrorMsg());
                            return false;
                        }
                    }else{
                        if (fieldValue.length() < validate.min()) {
                            System.out.println(validate.minErrorMsg());
                            return false;
                        }
                        if (fieldValue.length() > validate.max()) {
                            System.out.println(validate.maxErrorMsg());
                            return false;
                        }
                    }
                }
            }
        } catch (SecurityException e) {
            System.err.println("类型转换安全异常: " + e.getMessage());
        } catch (IllegalAccessException e) {
            System.err.println("访问属性改变错误: " + e.getMessage());
        } catch (IllegalArgumentException e) {
            System.err.println("非法参数: " + e.getMessage());
        }
        return true;
    }
}

  4.运行的代码

package valid;

public class Test {
    public static void main(String[] args) {
        User user = new User();

        user.setName("p");
//        user.setName("pppppppppppppppppppp");
//        user.setName("pff");
//        user.setAge("1");

        System.out.println(UserCheck.check(user));
    }
}

  5.运行结果

名字最小长度校验不通过!
false

参考:

https://www.cnblogs.com/liangweiping/p/3837332.html

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值