首页
学习
活动
专区
工具
TVP
发布
社区首页 >问答首页 >我可以更改方法参数的ConstraintValidator中的属性路径吗?

我可以更改方法参数的ConstraintValidator中的属性路径吗?
EN

Stack Overflow用户
提问于 2014-03-19 12:54:02
回答 2查看 11.5K关注 0票数 23

如果您熟悉Bean验证框架,就会知道无法获得方法参数的名称。因此,如果您对方法的第一个参数执行@NotNull约束,并且验证失败,则getPropertyPath将类似于"arg1“。

我想创建我自己的@NotNull版本,它可以接受一个值,例如@NamedNotNull( "emailAddress“)。但是我不知道如何覆盖我的Validator中的#getPropertyPath?有没有办法做到这一点,或者我坚持使用"arg1“或"arg2",等等。

编辑

根据我收到的答案,我提出了以下实现,允许我从@QueryParam或@PathParam批注中获取值,并将这些值用作@NotNull等Bean验证批注的属性路径。

对于Jersey,您需要创建以下类。注意DefaultParameterNameProvider的实现:

public class ValidationConfigurationContextResolver implements ContextResolver<ValidationConfig> {
    @Override
    public ValidationConfig getContext( final Class<?> type ) {
        final ValidationConfig config = new ValidationConfig();
        config.parameterNameProvider( new RestAnnotationParameterNameProvider() );
        return config;
    }

    static class RestAnnotationParameterNameProvider extends DefaultParameterNameProvider {

        @Override
        public List<String> getParameterNames( Method method ) {
            Annotation[][] annotationsByParam = method.getParameterAnnotations();
            List<String> names = new ArrayList<>( annotationsByParam.length );
            for ( Annotation[] annotations : annotationsByParam ) {
                String name = getParamName( annotations );
                if ( name == null )
                    name = "arg" + ( names.size() + 1 );

                names.add( name );
            }

            return names;
        }

        private static String getParamName( Annotation[] annotations ) {
            for ( Annotation annotation : annotations ) {
                if ( annotation.annotationType() == QueryParam.class ) {
                    return QueryParam.class.cast( annotation ).value();
                }
                else if ( annotation.annotationType() == PathParam.class ) {
                    return PathParam.class.cast( annotation ).value();
                }
            }

            return null;
        }
    }
}

然后,在您的RestConfig中,您需要添加以下行:

register( ValidationConfigurationContextResolver.class );

就这样。现在,您的ConstraintValidationExceptions将包含用来注释它们的QueryParam或PathParam的名称。例如:

 public void getUser( 
     @NotNull @QueryParam( "emailAddress" ) String emailAddress,
     @NotNull @QueryParam( "password" ) String password ) 
 { ... }
EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2017-05-31 05:37:00

Bean Validation 1.1引入了ParameterNameProvider接口,用于在创建违反约束的对象时为方法和构造函数参数提供名称。

Hibernate Validator5.2引入了ReflectionParameterNameProvider类,这是一个ParameterNameProvider实现,它使用反射来获取实际的参数名称(为了正常工作,它需要使用-parameters编译器参数编译类):

/**
 * Uses Java 8 reflection to get the parameter names.
 * <p>
 * <p>For this provider to return the actual parameter names, classes must be compiled with the '-parameters' compiler
 * argument. Otherwise, the JDK will return synthetic names in the form {@code arg0}, {@code arg1}, etc.</p>
 * <p>
 * <p>See also <a href="http://openjdk.java.net/jeps/118">JEP 118</a></p>
 *
 * @author Khalid Alqinyah
 * @since 5.2
 */
public class ReflectionParameterNameProvider implements ParameterNameProvider {

    @Override
    public List<String> getParameterNames(Constructor<?> constructor) {
        return getParameterNames(constructor.getParameters());
    }

    @Override
    public List<String> getParameterNames(Method method) {
        return getParameterNames(method.getParameters());
    }

    private List<String> getParameterNames(Parameter[] parameters) {

        List<String> parameterNames = newArrayList();
        for (Parameter parameter : parameters) {
            // If '-parameters' is used at compile time, actual names will be returned. Otherwise, it will be arg0, arg1...
            parameterNames.add(parameter.getName());
        }

        return parameterNames;
    }
}

Dropwizard对其进行了扩展,并使用JerseyParameterNameProvider添加了对JAX-RS @XxxParam注释的支持,这些注释也可以与其他JAX-RS实现一起使用:

/**
 * Adds jersey support to parameter name discovery in hibernate validator.
 * <p>
 * <p>This provider will behave like the hibernate-provided {@link ReflectionParameterNameProvider} except when a
 * method parameter is annotated with a jersey parameter annotation, like {@link QueryParam}. If a jersey parameter
 * annotation is present the value of the annotation is used as the parameter name.</p>
 */
public class JerseyParameterNameProvider extends ReflectionParameterNameProvider {

    @Override
    public List<String> getParameterNames(Method method) {
        Parameter[] parameters = method.getParameters();
        Annotation[][] parameterAnnotations = method.getParameterAnnotations();
        List<String> names = new ArrayList<>(parameterAnnotations.length);
        for (int i = 0; i < parameterAnnotations.length; i++) {
            Annotation[] annotations = parameterAnnotations[i];
            String name = getParameterNameFromAnnotations(annotations).orElse(parameters[i].getName());
            names.add(name);
        }
        return names;
    }

    /**
     * Derives member's name and type from it's annotations
     */
    public static Optional<String> getParameterNameFromAnnotations(Annotation[] memberAnnotations) {

        for (Annotation a : memberAnnotations) {
            if (a instanceof QueryParam) {
                return Optional.of("query param " + ((QueryParam) a).value());
            } else if (a instanceof PathParam) {
                return Optional.of("path param " + ((PathParam) a).value());
            } else if (a instanceof HeaderParam) {
                return Optional.of("header " + ((HeaderParam) a).value());
            } else if (a instanceof CookieParam) {
                return Optional.of("cookie " + ((CookieParam) a).value());
            } else if (a instanceof FormParam) {
                return Optional.of("form field " + ((FormParam) a).value());
            } else if (a instanceof Context) {
                return Optional.of("context");
            } else if (a instanceof MatrixParam) {
                return Optional.of("matrix param " + ((MatrixParam) a).value());
            }
        }

        return Optional.empty();
    }
}

如果你不使用Dropwizard,你可以使用上面的代码来创建你自己的实现。

泽西资源类/方法验证中使用的Validator的自定义可以使用ValidationConfig类并通过ContextResolver机制公开它:

public class ValidationConfigurationContextResolver 
        implements ContextResolver<ValidationConfig> {

    @Override
    public ValidationConfig getContext(final Class<?> type) {
        ValidationConfig config = new ValidationConfig();
        config.parameterNameProvider(new CustomParameterNameProvider());
        return config;
    }
}

然后在ResourceConfig中注册ValidationConfigurationContextResolver

有关更多详细信息,请参阅Jersey documentation about Bean Validation support

票数 8
EN

Stack Overflow用户

发布于 2014-03-20 04:54:23

如果您熟悉Bean验证框架,就会知道无法获取方法参数的名称

这是不太正确的。ParameterNameProvider验证指定了Bean的概念,允许您提供自己的实现。Hibernate Validator与ParaNamer集成以提供参数名称。有关详细信息,请参阅验证器online docs。一旦Validator支持Java 8,它也将支持Java 8参数命名工具。

国际海事组织,你应该试试ParaNamer。

票数 11
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/22496527

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档