访问注释实例附加到的方法以进行交叉参数验证

问题描述

我有一个Java注释的实例,该对象具有Method方法的目标:是否可以获取它所附加的Method对象?

我正在尝试进行跨参数验证,其中验证器将对参数组合应用逻辑以确定它们是否正确。但是,由于验证器可以应用于多种方法,因此我正在寻找一种方法标记哪些参数是哪个。

例如,在下面的示例中,我有一些方法使用两个自变量来指定范围的开头和结尾。我希望验证程序检查起始点是否大于结束点。我想在参数上使用注释来指示哪个参数是开始参数,哪个参数是结束参数,然后让验证器使用这些注释来确定在验证中使用哪些参数。

@Target(ElementType.ParaMETER)
@Retention(RetentionPolicy.RUNTIME)
public @interface StartParam {}

/***************************/

@Target(ElementType.ParaMETER)
@Retention(RetentionPolicy.RUNTIME)
public @interface EndParam {}

/***************************/

@Target({ElementType.CONSTRUCTOR,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = RangeParamsValidator.class)
@SupportedValidationTarget(ValidationTarget.ParaMETERS)
public @interface RangeParams { }

/***************************/

public class RangeParamsValidator implements ConstraintValidator<RangeParams,Object[]> {

    private int indexOfStartArg;
    private int indexOfEndArg;

    @Override
    public void initialize(
        RangeParams constraintAnnotation
    ) {
        // This is where I'm hoping to get the method that `constraintAnnotation`
        // is attached to,so I can iterate over its params and see which are
        // annotated with @StartParam and @EndParam so I can set indexOfStartArg
        // and indexOfEndArg;
    }

    @Override
    public boolean isValid(Object[] value,ConstraintValidatorContext context) {
        Integer start = value[indexOfStartArg];
        Integer end = value[indexOfEndArg];
        return start <= end;
    }
}


/***************************/

// Example use:

class Whatever {

    @RangeParams
    void doSomething(
        String ignoreThisArg,@StartParam int start,@FinishParam int finish,Object ignoreThisToo
    ) {
        // ...
    }
}

解决方法

我无法使用参数名称来解决这个问题,即使可能有办法。但是这里有一个关于参数位置的解决方案。

首先,您需要在约束中创建属性,以便可以传递字段的位置。

@Constraint(validatedBy = RangeParamsValidator.class)
@Target( {ElementType.CONSTRUCTOR,ElementType.METHOD} )
@Retention(RetentionPolicy.RUNTIME)
public @interface RangeParams {
    String message() default "Constraint violated";
    Class<?>[] groups() default {};
    Class<? extends Payload>[] payload() default {};
    int indexOfStartArg();
    int indexOfEndArg();
}

messagegroupspayload 符合标准(更多信息在 documentation 中)。重要的部分是提供索引的 2 个附加参数。如果您设置默认值,则参数将是可选的。

然后在验证器的初始化中您会得到值。

public class RangeParamsValidator implements ConstraintValidator<RangeParams,Object[]> {

private int indexOfStartArg;
private int indexOfEndArg;

@Override
public void initialize(RangeParams constraintAnnotation) {
    indexOfStartArg = constraintAnnotation.indexOfStartArg();
    indexOfEndArg = constraintAnnotation.indexOfEndArg();
}
...

您可以根据需要使用这些值。

然后提供您在注释中执行的值,如下所示:

@RangeParams(indexOfStartArg = 1,indexOfEndArg = 2)
void doSomething(
    String ignoreThisArg,int start,int finish,Object ignoreThisToo
) {
    // ...
}

我认为可以使用参数名称来实现,因为在 isValid() 上下文中包含参数名称列表。但是我只能通过接口的实现而不是通过接口本身来访问这些值,所以我需要将上下文转换为感觉不干净的实现。