当前位置: 首页 > 面试题库 >

Autowired在“自定义约束”验证器中给出Null值

方砚
2023-03-14
问题内容

我对Spring完全陌生,因此已经针对所问问题在SO上找到了一些答案。这里是链接:

我有一个Spring项目,我想在其中使用Hibernate Validator进行对象验证。根据我在网上阅读的内容和一些论坛,我尝试注入验证器,如下所示:

@Bean
  public Validator validator() {
    return new LocalValidatorFactoryBean().getValidator();
  }

但是无论我在哪里使用

@Autowired
Validator validator;

它采用了Spring的验证器实现,而不是Hibernate的验证器。我不知道如何准确地注入Hibernate验证程序,并简单地将其自动连接到其他类上,所以我使用了便宜的技巧,现在我的Java Config看起来像这样

@Bean
      public Validator validator() {
        // ValidatorImpl is Hibernate's implementation of the Validator
        return new ValidatorImpl();
      }

(如果有人能真正为我指出如何避免以这种Hacky方式获取Hibernate Validator的正确方向,我将不胜感激)

但是,让我们来谈谈主要问题:

这是自定义验证定义

@Target( { METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER } )
@Retention(RUNTIME)
@Constraint(validatedBy = EmployeeValidator.class)
@Documented
public @interface EmployeeValidation {
String message() default "{constraints.employeeConstraints}";
public abstract Class<?>[] groups() default {};
public abstract Class<? extends Payload>[] payload() default {};
}

我的自定义验证器

public class EmployeeValidator implements ConstraintValidator<EmployeeValidation , Object> {

@Autowired
private EmployeeService employeeService;

@Override
public void initialize(EmployeeValidation constraintAnnotation) {
//do Something
 }

@Override
public boolean isValid(String type, ConstraintValidatorContext context) {
    return false;
}
}

在上面的自定义约束验证器中,我获得了employeeService null。我知道在Spring启动时不会实例化ConstraintValidator的任何实现,但是我认为添加ValidatorImpl()可以真正解决该问题。但事实并非如此。

现在,我陷入了一个非常棘手的解决方法,并且我不想继续使用这样的代码。有人可以帮我解决我的情况吗?

PS这些是我在Java Config文件中的导入:

import org.hibernate.validator.HibernateValidator; 
import org.springframework.beans.factory.annotation.Autowired; 
import org.springframework.context.MessageSource; 
import org.springframework.context.annotation.Bean; 
import org.springframework.context.annotation.ComponentScan; 
import org.springframework.context.annotation.Configuration; 
import org.springframework.context.annotation.PropertySource; 
import org.springframework.context.support.ReloadableResourceBundleMessageSource; 
import org.springframework.core.env.Environment; 
import org.springframework.validation.Validator; 
import org.springframework.validation.beanvalidation.LocalValidatorFactoryBean; 
import org.springframework.web.servlet.LocaleResolver; 
import org.springframework.web.servlet.ViewResolver; 
import org.springframework.web.servlet.config.annotation.EnableWebMvc; 
import org.springframework.web.servlet.config.annotation.InterceptorRegistry; 
import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport; 
import org.springframework.web.servlet.i18n.CookieLocaleResolver; 
import org.springframework.web.servlet.i18n.LocaleChangeInterceptor; 
import org.springframework.web.servlet.view.InternalResourceViewResolver; 

问题答案:

我希望该解决方案可以帮助:

@Bean
public Validator validator () {

    ValidatorFactory validatorFactory = Validation.byProvider( HibernateValidator.class )
        .configure().constraintValidatorFactory(new SpringConstraintValidatorFactory(autowireCapableBeanFactory))
        .buildValidatorFactory();
    Validator validator = validatorFactory.getValidator();

    return validator;
}

使用初始化验证器,SpringConstraintValidatorFactory以便注入能够工作,并将验证器实现提供为Hibernate.class的工作方式如下:

  1. 你选择的库将验证你的对象
  2. 你的自定义验证器将能够使用Spring的功能,同时让Hibernate执行验证。

工作原理:除非被调用,否则Hibernate ConstraintValidatorFactory不会初始化任何东西ConstraintValidators,而是SpringConstraintValidatorFactory通过给予AutowireCapableBeanFactory它来初始化。

编辑

如@shabyasaschi的评论之一所述,要注入,autowireCapableBeanFactory你可以将方法签名更改为:

Validator validator(final AutowireCapableBeanFactory autowireCapableBeanFactory) {

或在配置文件中为其添加getter和setter,如下所示:

public AutowireCapableBeanFactory getAutowireCapableBeanFactory() {
        return autowireCapableBeanFactory;
}

public void setAutowireCapableBeanFactory(AutowireCapableBeanFactory autowireCapableBeanFactory) {
        this.autowireCapableBeanFactory = autowireCapableBeanFactory;
}


 类似资料:
  • 我对Spring是完全陌生的,对于所问的问题,我已经在SO上寻找了一些答案。以下是链接: Spring 3.1 Autowiring不能在自定义约束验证器中工作 它采用了Spring的验证器实现,而不是Hibernate的验证器。我不知道如何准确地注入Hibernate验证器,并简单地在其他类之间自动调用它,所以我使用了一个廉价的技巧,现在我的Java配置如下所示 (如果有人能告诉我如何避免以这种

  • 我有一个基于Spring的WebApp。我在控制器中使用了几个带有注释@repository和@transactional的存储库类。那部分工作得很好。 我创建了一个自定义约束验证器,它必须访问存储库。现在我不明白为什么存储库是空的。我尝试用@component注释来注释验证器。包含所有这些类的基本包位于XML部分。那么,我还应该做些什么来确保存储库依赖项注入的工作。这就是我的约束验证器的样子。

  • 我已经搜索并找到了很多关于这个问题的话题,但无论出于什么原因,解决方案都不能解决我的问题。 我运行的是Spring boot 1.5.8和Hibernate。 我创建了一个自定义验证注释,以检查电子邮件与数据库相比是否唯一。注释使用查询数据库,以检查数据库中是否已存在电子邮件。我一直得到一个异常,bean为空。 我的理论是使用了hibernate验证器,而不是< code > UniqueEmai

  • 我在项目中使用bean验证,我想为现有的约束注释编写一个自定义验证器。 例如,我有一个类,它表示一个名为CustomDateTime的日期/时间。在使用此类作为例如出生日期的类中,我想用过去的日期对字段进行注释: 然后,我通过实现ConstraintValidator创建一个自定义验证器 我知道您通常会创建这样的单独注释: 但对我来说,这似乎是双重代码;-) 如何注册要与一起使用的自定义验证器?

  • 我有一个像下面这样的模型对象,带有自定义约束验证器。自定义验证器检查是否填充了fileName或小时。 有一种方法将此作为输入,它验证所有以下条件 > 条件不为空(通过默认验证器) criteria.id不为空(通过默认验证器) criteria.name不为空(通过默认验证器) 标准文件名或小时不为空(通过自定义验证器) 空评估(@NotNull@有效标准标准){} 现在,当我为这个模型类编写单

  • 我有一个自定义注释@UniqueModel,它由ConstraintValidator验证: 问题是,我需要在调用存储库的safe()-方法之前进行验证,否则字段注入将无法工作。 因此,我创建了一个带有@Valid注释的委托方法,以便在以下情况之前强制进行唯一验证: 不幸的是,这不起作用,似乎@Valid注释被Spring忽略了。 我如何确保验证的正确时间?