Java运行时注解

标签 java dynamic annotations runtime

存在以下类,它由描述可能的整个数据库的预定义 UUID 组成。

public class Predefined {
    @NotNull
    @Size(min = 1, max = 25)
    public UUID phone = UUID.fromString("47b58767-c0ad-43fe-8e87-c7dae489a4f0");

    @NotNull
    @Size(min = 1, max = 40)
    public UUID company = UUID.fromString("f9a1e8f4-b8c0-41f2-a626-49c11da8d5c2");
}

这些值通过 Web 服务作为 key 对值接收:然后将它们放入 HashMap 中。

47b58767-c0ad-43fe-8e87-c7dae489a4f0 = +00112233445566778899

f9a1e8f4-b8c0-41f2-a626-49c11da8d5c2 = 某个虚拟公司名称

当我收到一个 UUID 时,我知道我正在创建预定义类的实例,然后获取预定义类中字段的注释,即:

Annotation[] annon = field.getDeclaredAnnotations(); 

现在我需要在运行时再次检查这些注释从 Web 服务获取的值,即“+00112233445566778899”和“someVirtualCompnayName”

这可能吗? 我对涉及 JSR 303 的示例特别感兴趣。

为什么我有这样的结构: DAO 、 @Repository 类具有不同的结构,即

联系方式

联系人属性

联系人属性类型

其中数据库“contact_attibute_type”表示“公司”和“电话”。第二个表,即“contact_attrbute”代表“company”和“phone”的实际值。

现在我需要一种方法来验证这些值,然后再将它们写入 hibernate 状态,因此我得到“公共(public) UUID 电话”,然后尝试将这些约束应用于我从用户那里获得的实际值,即“+00112233445566778899” .

最佳答案

我将发布我用来验证您的测试用例的完整代码(包括一个简单的可执行演示):

注释:

package annotations;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target( {ElementType.FIELD })
@Retention(RetentionPolicy.RUNTIME)
public @interface NotNull
{

}

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target( {ElementType.FIELD })
@Retention(RetentionPolicy.RUNTIME)
public @interface Size
{
    int min() default 0;
    int max();
}

预定义类:

public class Predefined
{
    @NotNull
    @Size(min = 1, max = 25)
    public UUID phone;

    @NotNull
    @Size(min = 1, max = 40)
    public UUID company;

    public Predefined(UUID phone, UUID company)
    {
        this.phone = phone;
        this.company = company;
    }
}

validator 类迭代声明的字段并检查它们的注释和字段/值映射:

public class PredefinedValidator
{
    public boolean validate(Predefined predefined, Map<UUID, String> mappings)
    {
        if (predefined == null)
            return false;

        for (Field field :predefined.getClass().getDeclaredFields())
        {
            if (field.getType().equals(UUID.class))
            {
                try
                {
                    Annotation[] annotations = field.getDeclaredAnnotations();
                    UUID uuid = (UUID)field.get(predefined);
                    if (!this.validateField(uuid, annotations, mappings))
                        return false;
                }
                catch (IllegalArgumentException | IllegalAccessException ex)
                {
                    Logger.getLogger(PredefinedValidator.class.getName()).log(Level.SEVERE, null, ex);
                }
            }
        }
        return true;
    }

    private boolean validateField(UUID field, Annotation[] annotations, Map<UUID, String> mapping)
    {
        boolean containsSize = false;
        boolean containsNotNull = false;
        int minSize = -1;
        int maxSize = -1;

        // fetch which annotations are available for the provided field
        for (Annotation annotation : annotations)
        {
            if (annotation instanceof Size)
            {
                containsSize = true;
                Size size = (Size)annotation;
                minSize = size.min();
                maxSize = size.max();
            }
            else if (annotation instanceof NotNull)
                containsNotNull = true;
        }

        // check if the provided value is null and an annotatition for @NotNull
        // is set
        if (field == null && containsNotNull)
            return false;

        if (containsSize)
        {
            // get the value of the mapped UUID which we are going to validate
            String value = mapping.get(field);
            if (value == null && containsNotNull)
                return false;
            else if (value == null)
                return true;

            // check if the length of the value matches
            if (value.length() <= minSize || value.length() >= maxSize)
                return false;
        }

        // passed all tests
        return true;
    }
}

最后但并非最不重要的一个简单演示:

public static void main(String ... args)
{
    Map<UUID, String> mappings = new HashMap<>();
    mappings.put(UUID.fromString("47b58767-c0ad-43fe-8e87-c7dae489a4f0"), "+00112233445566778899");
    mappings.put(UUID.fromString("f9a1e8f4-b8c0-41f2-a626-49c11da8d5c2"), "someVirtualCompnayName");       

    Predefined predefined = new Predefined(
            UUID.fromString("47b58767-c0ad-43fe-8e87-c7dae489a4f0"), 
            UUID.fromString("f9a1e8f4-b8c0-41f2-a626-49c11da8d5c2"));
    Predefined predefined2 = new Predefined(
            UUID.randomUUID(), 
            UUID.fromString("f9a1e8f4-b8c0-41f2-a626-49c11da8d5c2"));
    Predefined predefined3 = new Predefined(
            null, 
            UUID.fromString("f9a1e8f4-b8c0-41f2-a626-49c11da8d5c2"));
    PredefinedValidator validator = new PredefinedValidator();

    System.out.println("predefined is valid: "+validator.validate(predefined, mappings));
    System.out.println("predefined is valid: "+validator.validate(predefined2, mappings));
    System.out.println("predefined is valid: "+validator.validate(predefined3, mappings));

    mappings.put(UUID.fromString("f9a1e8f4-b8c0-41f2-a626-49c11da8d5c2"), "someVirtualCompnayNamesomeVirtualCompnayNamesomeVirtualCompnayNamesomeVirtualCompnayName"); 
    System.out.println("predefined is valid: "+validator.validate(predefined, mappings));
}

HTH

关于Java运行时注解,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21229156/

相关文章:

dynamic - 站点地图文件是否有最大大小?

c++ - 使用指针创建二维动态数组

python - 如何在方法签名中注释指向C字符数组的指针?

java - 我可以注释从父类(super class)继承的成员吗?

java - Kongchen Swagger 忽略了 @ApiModelProperty 注释

java - Java MapReduce程序

java - 无法确定何时抓取框架边框

java - 以 double 移动小数位

PHP:动态或编程捕获 block

java - 将子类型中类型参数的注释链接到父类(super class)型中声明类型参数