Hibernate Validator 校验 (二)

还想看更多文章的朋友可以访问我的个人博客


Hibernate Validator 校验 (一)中,谈了如何利用 Hibernate Validator 提供的约束注解完成较为简单的对象校验,但这也不能满足开发的需求。这时,我们需要自己实现校验逻辑。

自定义校验逻辑

自实现校验有很多优点,一个项目下,肯定存在多个服务对同一类型的实例操作,而其校验逻辑大多类似。通过自实现校验将该逻辑抽象出来,一方面减少代码重复,又可以满足当下模块开发的思想。

先来看看测试类与 Controller 层实现,其实与上一篇中的实现无太大差别,只是此次我们将使用正则表达式校验username,规则是匹配任何字类字符,包括下划线,因此此处我们模拟发起增添用户的POST请求。代码如下:

Controller层实现:

@RestController
@RequestMapping("/user")
public class UserController {
  @PostMapping
  public User updateUser(@PathVariable Integer id, @RequestBody @Valid User user, BindingResult result) {
      // 输出错误信息
      if (result.hasErrors()) {
          result.getAllErrors().forEach(err -> System.out.println(err.getDefaultMessage()));
      }

      user.setId(id);

     return userService.update(user);
  }
}

测试类:

@RunWith(SpringRunner.class)
@SpringBootTest
public class UserControllerTest {

  @Autowired
  private WebApplicationContext wac;

  private MockMvc mockMvc;

  @Before
  public void setUp() {
      this.mockMvc = MockMvcBuilders.webAppContextSetup(wac).build();
  }

  @Test
  public void testValidUser() throws Exception {
      Date date = new Date();
      String content = "{\"username\": \"@ksjkd\",\"password\": \"password\",\"birthday\":"
           - date.getTime() + "}";
      mockMvc.perform(post("/user")
                      .contentType(MediaType.APPLICATION_JSON_UTF8)
                      .content(content))
                      .andExpect(status().isOk());
  }

}

上一篇中也有提到,JPA Validation 规范对约束的定义包括两部分,一是约束注解,如上篇中用到的@NotBlankNotNull等就是约束注解;二是约束校验器,每一个约束注解都存在对应的约束校验器,约束校验器实现具体的校验逻辑。我们自己实现 validator 也需要满足这两点。

一、创建约束注解与其对应的约束校验器

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = { TestValidator.class })
public @interface TestConstraint {

    String message() default ""; // 约束注解校验时的输出消息

    Class<?>[] groups() default { }; // 约束注解在校验时所属的组别

    Class<? extends Payload>[] payload() default { }; // 约束注解的有效负载
}

如上,在该注解上@Target@Retention是 Java 中元注解中的两个,分别用于指明该注解的作用目标与保留位置,此处不多做赘述。而Constraint注解特用于指明 JPA 约束注解与约束校验器的对应关系。

另外,JPA Validation 规范要求自实现的约束注解必须声明以上三个参数,分别是:message, groups, payload。其中message属性便是当校验器校验失败时的输出消息,另两个属性的大致用途已在代码中注释,不再详述。

如下,为该校验器的实现:

public class TestValidator implements ConstraintValidator<TestConstraint, String> {

    @Autowired
    private TestService testService;

    @Override
    public void initialize(TestConstraint constraintAnnotation) {
        // 添加约束注解在初始化时需要做的动作
    }

    @Override
    public boolean isValid(String value, ConstraintValidatorContext context) {
        System.out.println(testService.test(value));

        String pattern = "^\\w+$"; // 匹配任何字类字符,包括下划线。与"[A-Za-z0-9_]"等效

        Pattern r = Pattern.compile(pattern);
        Matcher m = r.matcher(value);

        return m.matches();
    }
}

自实现的校验器必须实现ConstraintValidator<A, T>接口,该接口有两个泛型,分别是该校验器所“对应的约束注解的类类型”和“被校验对象的类类型”。这里是针对String类型的username字段进行校验,所以两泛型类型分别为TestConstraintString

如果是 Spring 项目,当实现该接口则会自动被 Bean 容器收集为 Bean。因此,我们可以按需求Autowired需要的BeanTestService实现如下:

@Service
public class TestService {
    public String test(String o) {
        return "test:[" + o + "]";
    }
}

本例中,该校验器采用“正则规则”对字符串进行匹配。

二、在实体中声明需要校验的字段

@Entity
public class User {

    @TestConstraint(message = "测试校验:用户名必须是数字、字母或_的组合")
    private String username;

    ...
}

利用我们方才定义的约束注解@TestConstraintusername 字段进行校验,并自定义校验错误消息。

测试结果

在测试请求中,我们模拟的请求数据中username字段中添加了非法字符@,因此校验肯定是无法通过的:

image

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,402评论 6 499
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,377评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,483评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,165评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,176评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,146评论 1 297
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,032评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,896评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,311评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,536评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,696评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,413评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,008评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,659评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,815评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,698评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,592评论 2 353

推荐阅读更多精彩内容