3

我正在使用HibernateValidator 4.3.1。验证在整个应用程序中按预期执行。

我已经注册了一些自定义编辑器来执行全局验证,例如确保文本字段中的数值(doubleint),确保有关Joda-Time API 的有效日期等。

在这种类型的验证中,我允许空值/空值,方法是像往常一样将allowEmpty参数设置false为单独验证它,特别是当这些字段留空时显示单独的用户友好错误消息。

因此,除了使用 HibernateValidator 和自定义编辑器进行验证之外,我还尝试使用以下验证策略。同样,这种验证仅适用于那些为自定义编辑器注册的字段在留空时

下面是实现org.springframework.validation.Validator接口的类。

package test;

import org.springframework.stereotype.Component;
import org.springframework.validation.Errors;
import org.springframework.validation.ValidationUtils;
import org.springframework.validation.Validator;
import validatorbeans.TempBean;

@Component
public final class TempValidator implements Validator {

    @Override
    public boolean supports(Class<?> clazz) {
        System.out.println("supports() invoked.");
        return TempBean.class.isAssignableFrom(clazz);
    }

    @Override
    public void validate(Object target, Errors errors) {
        TempBean tempBean = (TempBean) target;

        System.out.println("startDate = " + tempBean.getStartDate() + " validate() invoked.");
        System.out.println("doubleValue = " + tempBean.getDoubleValue() + " validate() invoked.");
        System.out.println("stringValue = " + tempBean.getStringValue() + " validate() invoked.");

        ValidationUtils.rejectIfEmptyOrWhitespace(errors, "startDate", "java.util.date.nullOrEmpty.error");
        ValidationUtils.rejectIfEmptyOrWhitespace(errors, "doubleValue", "java.lang.double.nullOrEmpty.error");
    }
}

该类是用@Component注解指定的,以便它可以自动连接到特定的 Spring 控制器类。调试语句完全根据用户提供的输入显示。

以下是控制器类。

package controller;

import customizeValidation.CustomizeValidation;
import java.util.Map;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.validation.Valid;
import javax.validation.groups.Default;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.validation.BindingResult;
import org.springframework.validation.DataBinder;
import org.springframework.web.bind.annotation.ModelAttribute;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import test.TempValidator;
import validatorbeans.TempBean;

@Controller
public final class TempController {

    @Autowired
    private TempService tempService;

    private TempValidator tempValidator;

    public TempValidator getTempValidator() {
        return tempValidator;
    }

    @Autowired
    public void setTempValidator(TempValidator tempValidator) {
        this.tempValidator = tempValidator;
    }

    @RequestMapping(method = {RequestMethod.GET}, value = {"admin_side/Temp"})
    public String showForm(@ModelAttribute("tempBean") @Valid TempBean tempBean, BindingResult error, Map model, HttpServletRequest request, HttpServletResponse response) {
        return "admin_side/Temp";
    }

    @RequestMapping(method = {RequestMethod.POST}, value = {"admin_side/Temp"})
    public String onSubmit(@ModelAttribute("tempBean") @Valid TempBean tempBean, BindingResult errors, Map model, HttpServletRequest request, HttpServletResponse response) {
        //tempValidator.supports(TempBean.class);
        //tempValidator.validate(tempBean, errors);

        DataBinder dataBinder = new DataBinder(tempBean);
        dataBinder.setValidator(tempValidator);
        dataBinder.validate();

        //errors=dataBinder.getBindingResult();
        if (CustomizeValidation.isValid(errors, tempBean, TempBean.ValidationGroup.class, Default.class) && !errors.hasErrors()) {
            System.out.println("Validated");
        }

        return "admin_side/Temp";
    }
}

我从 Spring 控制器类本身(我确实想要)调用验证器

DataBinder dataBinder = new DataBinder(tempBean);
dataBinder.setValidator(tempValidator);
dataBinder.validate();

调用了验证器,但未执行预期的验证。

如果我只使用以下语句(上面已注释掉)手动调用验证器,

tempValidator.validate(tempBean, errors);

然后执行验证。所以我不相信我的验证器工作正常。为什么它无法使用DataBinder

在我的application-context.xml文件中,这个 bean 简单地配置如下。

<bean id="tempValidator" class="test.TempValidator"/>

自动检测到以下许多包,包括包含该类的test包。TempValidator

<context:component-scan base-package="controller spring.databinder validatorbeans validatorcommands test" use-default-filters="false">
    <context:include-filter expression="org.springframework.stereotype.Controller" type="annotation"/>
    <context:include-filter expression="org.springframework.web.bind.annotation.ControllerAdvice" type="annotation"/>  
</context:component-scan>

我什至试图把

<bean id="validator" class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean"/>

在我的dispatcher-servlet.xml档案中。

我在这里俯瞰什么?

4

2 回答 2

4

如果我很好地理解您尝试实现的目标 - 区分空白字段和​​输入的错误值 - 您可以使用更简单的方法:

public class MyBean {

    @NotNull
    @DateTimeFormat(pattern="dd.MM.yyyy HH:mm") 
    private DateTime date;

    @NotNull
    @Max(value=5)
    private Integer max;

    @NotNull
    @Size(max=20)
    private String name;

    // getters, setters ... 
} 

控制器映射:

public void submitForm(@ModelAttribute @Valid MyBean myBean, BindingResult result) {

    if (result.hasErrors){
       // do something}
    else{
       // do something else
    }        
}

验证消息:

NotNull=Required field.
NotNull.date=Date is required field.
NotNull.max=Max is required field.
Size=Must be between {2} and {1} letters.
Max=Must be lower than {1}.
typeMismatch.java.lang.Integer=Must be number.
typeMismatch.org.joda.time.DateTime=Required format dd.mm.yyyy HH:mm

弹簧配置:

@Configuration
public class BaseValidatorConfig {

    @Bean
    public LocalValidatorFactoryBean getValidator() {

        LocalValidatorFactoryBean lvfb = new LocalValidatorFactoryBean();
        lvfb.setValidationMessageSource(getValidationMessageSource());
        return lvfb;
    }

    protected MessageSource getValidationMessageSource() {// return you validation messages ...}
}

如果需要,我可以提供更多细节和解释。

于 2013-02-27T07:58:32.767 回答
0

我不知道为什么问题中提到的方法不起作用。我没有让它工作,但是浏览了这个文档,我发现了另一种根据我的要求对我有用的方法。

@InitBinder我在注释指定的方法中设置了验证器。

来自文档

遇到 @Valid 方法参数时调用的 Validator 实例可以通过两种方式进行配置。首先,您可以在 @Controller 的 @InitBinder 回调中调用 binder.setValidator(Validator)。这允许您为每个 @Controller 类配置一个 Validator 实例:

具体来说,在我的要求中,仅应在更新或将数据插入数据库时​​执行验证,即当按下这些操作的关联提交按钮时(在我的应用程序中,这两个任务(插入和更新)都有一个通用按钮)谁的名字是btnSubmit)。

在任何其他情况下(例如,按下删除按钮时),验证应该被静音。为了满足这个要求,我注册了验证器如下。

@InitBinder
protected void initBinder(WebDataBinder binder, WebRequest webRequest) {
    if (webRequest.getParameter("btnSubmit") != null) {
        binder.setValidator(new TempValidator());
    } else {
        binder.setValidator(null);
    }
}

在这种情况下,validator -TempValidator只会btnSubmit在客户端单击其名称属性的提交按钮时设置。

无需在任何地方进行 xml 配置以及自动装配。

示例控制器类现在如下所示。

@Controller
public final class TempController {

    @Autowired
    private TempService tempService;

    @InitBinder
    protected void initBinder(WebDataBinder binder, WebRequest webRequest) {
        if (webRequest.getParameter("btnSubmit") != null) {
            binder.setValidator(new TempValidator());
        } else {
            binder.setValidator(null);
        }
    }

    //Removed the @Valid annotation before TempBean, since validation is unnecessary on page load.
    @RequestMapping(method = {RequestMethod.GET}, value = {"admin_side/Temp"})
    public String showForm(@ModelAttribute("tempBean") TempBean tempBean, BindingResult error, Map model, HttpServletRequest request, HttpServletResponse response) {
        return "admin_side/Temp";
    }

    @RequestMapping(method = {RequestMethod.POST}, value = {"admin_side/Temp"})
    public String onSubmit(@ModelAttribute("tempBean") @Valid TempBean tempBean, BindingResult errors, Map model, HttpServletRequest request, HttpServletResponse response) {
        if (CustomizeValidation.isValid(errors, tempBean, TempBean.ValidationGroup.class, Default.class) && !errors.hasErrors()) {
            System.out.println("Validated");
        }
        return "admin_side/Temp";
    }
}

方法中的WebRequest参数initBinder()并不意味着处理整个 Http 请求。它仅用于使用通用请求元数据。

Javadocs 关于WebRequest.

Web 请求的通用接口。主要用于通用 Web 请求拦截器,使它们能够访问通用请求元数据,而不是用于实际处理请求。

如果我可能会遵循某些错误,请澄清它或添加另一个答案。

于 2013-02-21T18:29:29.390 回答