SpringBoot 如何进行业务校验,老鸟们都这么玩的~

大家好,我是飘渺。

今天继续给大家带来 SpringBoot老鸟系列 的第七篇,来聊聊在SpringBoot项目中如何实现业务异常校验Assert。

希望通过今天的文章,咱们能够了解到:

  1. 如何使用Assert参数校验?

  2. 为什么用了Validator参数校验,还必须再用Assert参数校验?

首先我们来看看为什么需要Assert?

为什么需要Assert?

Assert翻译为中文为"断言",它是spring的一个util类,org.springframework.util.Assert一般用来断定某一个实际的值是否为自己预期想得到的,如果不一样就抛出异常。

在前面的文章中我们详细介绍了SpringBoot中如何集成参数校验Validator,那既然有了Validator为什么还需要Assert呢?

主要原因有两个:

  1. 因为Validator只解决了参数自身的数据校验,解决不了参数和业务数据之间校验。

    例如以下代码,Validator是搞不定的。

public void test1(int accountId) {
    Account account = accountDao.selectById(accountId);
    if (account == null) {
        throw new IllegalArgumentException("用户不存在!");
    }
}
  1. 采用Assert能使代码更优雅,更简洁。

    还是上面的例子,如果采用Assert可以这样写:

public void test2(int accountId) {
    Account account = accountDao.selectById(accountId);
    Assert.notNull(account, "用户不存在!");
}

如何使用Assert?

在SpringBoot中使用Assert非常简单,直接使用Assert提供的静态方法即可。

@RestController
@RequestMapping("assert")
@Slf4j
public class AssertController {

    @DeleteMapping("/user/{id}")
    public void deleteUser(@PathVariable("id") String id) {
        //模拟数据库查询用户
        UserVO user = getUserById(id);
        Assert.notNull(user, "用户不存在!");
    }

    private UserVO getUserById(String id) {
        return null;
    }
}

如上,AssertController有一个删除用户的接口,当删除用户时我们需要先校验用户是否存在。这里直接使用Assert.notNull()进行UserVO的非空校验。

此时访问接口,返回的json对象如下:

{
  "timestamp": "2022-01-10T14:17:13.335+00:00",
  "status": 500,
  "error": "Internal Server Error",
  "message": "",
  "path": "/assert/user/javadaily"
}

从测试结果来看,assert抛出的异常是springboot原生json对象,很明显我们必须将其加入全局异常拦截器RestExceptionHandler

加入全局异常拦截器

查看Assert.notNull()方法,可以看到Assert抛出的是IllegalArgumentException异常,所以我们只需要在全局异常拦截器中加入IllegalArgumentException拦截即可。

/**
  * Assert异常
  */
@ExceptionHandler(IllegalArgumentException.class)
@ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
public ResultData<String> exception(IllegalArgumentException e) {
  return ResultData.fail(ReturnCode.ILLEGAL_ARGUMENT.getCode(),e.getMessage());
}

此时再次访问接口,返回的数据结果为:

{
  "status": 3001,
  "message": "用户不存在!",
  "data": null,
  "timestamp": 1641825258876
}

符合结果预期。

常见的Assert使用场景

  • 逻辑断言

  1. isTrue()如果条件为假抛出IllegalArgumentException 异常。

  2. state()该方法与isTrue一样,但抛出IllegalStateException异常。

  • 对象和类型断言

  1. notNull()通过notNull()方法可以假设对象不null:

  2. isNull()用来检查对象为null:

  3. isInstanceOf()使用isInstanceOf()方法检查对象必须为另一个特定类型的实例

  4. isAssignable()使用Assert.isAssignable()方法检查类型

  • 文本断言

  1. hasLength()如果检查字符串不是空符串,意味着至少包含一个空白,可以使用hasLength()方法。

  2. hasText()我们能增强检查条件,字符串至少包含一个非空白字符,可以使用hasText()方法。

  3. doesNotContain()我们能通过doesNotContain()方法检查参数不包含特定子串。

  • Collection和map断言

  1. Collection应用notEmpty()如其名称所示,notEmpty()方法断言collection不空,意味着不是null并包含至少一个元素。

  2. map应用notEmpty()同样的方法重载用于map,检查map不null,并至少包含一个entry(key,value键值对)。

  • 数组断言

  1. notEmpty()notEmpty()方法可以检查数组不null,且至少包括一个元素:

  2. noNullElements()noNullElements()方法确保数组不包含null元素

小结

Assert断言,可以替换传统的if判断,大量减少业务参数校验的代码行数,提高程序的可读性,这种风格是目前比较流行的方式。

tips : 老鸟系列源码已经上传至GitHub,需要的关注本公众号并回复关键字 0923 获取源码地址。

  • 2
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

飘渺Jam

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值