【springboot】 通过jdk8特性 优雅的写入日志(非常推荐) 解决@valid失效问题

前言

@valid 可以帮助我们节省很多代码 比较方便 但操作失误时 可能会失效 达不到我们预期效果;

@valid会有个问题 因为注解过于方便 反而会导致拦截后 错误日志的收集会比较麻烦 ,以及在面对有时需要拦截 有时不需要拦截的特定场景下 显得无计可施 此时我们还是要回归手写校验

本文将从解决@valid失效问题 及提供更灵活的方案两个方面进行简述

@valid失效问题

  1. 检查依赖 单独引入validation-api可能无效 我们看看springboot给出的提示
    在这里插入图片描述
		<!-- version 和springboot版本保持一致即可 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-validation</artifactId>
        </dependency>
  1. 检查是否使用错误

需要校验的类添加@Validated注解(有接口的情况下 可加在接口 也可在实现类 推荐统一加在接口)

方法参数需要添加@Valid注解 (有接口的情况下 需要加在接口的方法参数)

实体类参数需要添加校验注解 如@NotNull 下面用代码举个例子

// 注意是 interface 
@Validated
public interface DeviceConfigService extends IService<DeviceConfigDO> {

    void putDeviceConfigList(@Valid DeviceConfigUpdaterCmd cmd);
}
@Data
@EqualsAndHashCode
public class DeviceConfigUpdaterCmd implements Serializable {

    private static final long serialVersionUID=1L;

    @NotNull(message = "错误")
    private Long id;
   }

替代方案

举个例子 我们项目和外部的接口进行对接 鉴于主动防御原则 我们可能对其某些参数进行校验,当id参数为空时 我们需要写入日志 方便快速定位(甩锅)问题

传统的写法是

if (null != cmd.getId){
  throw new RuntimeException("参数不能为空");
}

你是否也感觉一片的 if 看着很让人恼火 又无计可施呢?
再后来 可能我们会从spring源码中的断言类受到启发,用spring的姿势是这么去判断的

Assert.notNull(id,"不能为空");
	// spring的 Assert 类
    public static void notNull(@Nullable Object object, String message) {
        if (object == null) {
            throw new IllegalArgumentException(message);
        }
    }

但这随之而来有个很大的问题 ,我们如何打日志呢?

这个时候我们应该想到 jdk8开始 提供了函数式接口,我们可以直接将方法传入

比如Function Consumer 都可以做到,但是关键在于方法传入 如何让它执行呢?

所以我们选择了Consumer消费型函数,看完下面代码 我们就能很好的理解为什么它叫消费型函数了:
在这里插入图片描述

如果对Consumer不熟悉的同学 或许会感到疑惑 网上似乎都是此类教程,这到底有什么用?接着往下看 我们用函数式简化一下

在这里插入图片描述

我们能看出来,我们test()方法的参数 也是一个方法,并且accept会将该方法执行(消费),
且accept(T t)是必须要有参数的 ,

在这里插入图片描述

然而 我们写日志的时候 log.error(“msg”) ,这一步骤就够了 我们并不需要额外将 "msg” 交给accpet来处理,
那应该怎么办呢?

于是我们可以参数给定一个空串 :

	// 自定义一个CommonAssert类
    public static void notNull(@Nullable Object object, String msg, Consumer consumer) {
        if (null == object) {
        	// 消费该方法 (执行该方法)
            consumer.accept("");
            throw new RuntimeException(msg);
        }
    }

使用方代码:

        CommonAssert.notNull(id, "id error", item -> log.error("id error"));

优化:

lambada表达式中 item指代的是accept(xx)里面的参数xx, 我们写入日志就变成了 item -> log.error(item)
简写 log::error

    public static void notNull(@Nullable Object object, String msg, Consumer<String> consumer) {
        if (null == object) {
        	// 消费该方法 (执行该方法)
            consumer.accept(msg);
            throw new RuntimeException(msg);
        }
    }

使用方代码:

        CommonAssert.notNull(id, "id error", log::error);
        // 或自定义日志
        CommonAssert.notNull(id, "id error", item -> log.error("your error msg"));

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

孟秋与你

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

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

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

打赏作者

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

抵扣说明:

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

余额充值