前言
项目中参数校验十分重要,它可以保护我们应用程序的安全性和合法性。我想大家通常的做法是像下面这样做的:
@Override
public void validate(SignUpCommand command) {
validateCommand(command); // will throw an exception if command is not valid
validateUsername(command.getUsername()); // will throw an exception if username is duplicated
validateEmail(commend.getEmail()); // will throw an exception if email is duplicated
}
这么做最大的优势就是简单直接,但是如果验证逻辑很复杂,会导致这个类变得很庞大,而且上面是通过抛出异常来改变代码执行流程,这也是一种不推荐的做法。
那么有什么更好的参数校验的方式呢?本文就推荐一种通过责任链设计模式来优雅地实现参数的校验功能,我们通过一个用户注册的例子来讲明白如何实现。
- 有效的注册数据——名字、姓氏、电子邮件、用户名和密码。
- 用户名必须是唯一的。
- 电子邮件必须是唯一的。
定义用户注册和验证结果类
- 定义一个
SignUpCommand
类用来接受用户注册的属性信息。并且使用@Value
注解让这个类不可变。
import lombok.Value;
import javax.validation.constraints.*;
@Value
public class SignUpCommand {
@Min(2)
@Max(40)
@NotBlank
private final String firstName;
@Min(2)
@Max(40)
@NotBlank
private final String lastName;
@Min(2)
@Max(40)
@NotBlank
private final String username;
@NotBlank
@Size(max = 60)
@Email
private final String email;
@NotBlank
@Size(min = 6, max = 20)
private final String rawPassword;
- 使用
javax.validation
中的注解如@NotBlank
、@Size
来验证用户注册信息是否有效。 - 使用
lombok
的注解@Value
,因为我希望命令对象是不可变的。注册用户的数据应与注册表中填写的数据相同。
- 定义存储验证结果类
ValidationResult
,如下所示:
@Value
public class ValidationResult {
private final boolean isValid;
private final String errorMsg;
public static ValidationResult valid() {
return new ValidationResult(true, null);
}
public static ValidationResult invalid(String errorMsg) {
return new ValidationResult(false, errorMsg);
}
public boolean notValid() {
return !isValid;
}
}
- 在我看来,这是一种非常方便的方法返回类型,并且比抛出带有验证消息的异常要好。
- 既然是责任链,还需要定义一个“链”类
ValidationStep
,它是这些验证步骤的超类,我们希望将它们相互“链接”起来。
public abstract class ValidationStep
核心验证逻辑
现在我们开始进行参数校验的核心逻辑,也就是如何把上面定义的类给串联起来。
- 我们定义一个用于注册验证的接口类
SignUpValidationService
public interface SignUpValidationService {
ValidationResult validate(SignUpCommand command);
}
- 现在我们可以使用上面定义的类和责任链模式来轻松的实现,代码如下:
import lombok.AllArgsConstructor;
import org.springframework.stereotype.Service;
import javax.validation.ConstraintViolation;
import javax.validation.Validation;
import javax.validation.Validator;
import javax.validation.ValidatorFactory;
import java.util.Set;
@Service
@AllArgsConstructor
public class DefaultSignUpValidationService implements SignUpValidationService {
private final UserRepository userRepository;
@Override
public ValidationResult validate(SignUpCommand command) {
return new CommandConstraintsValidationStep()
.linkWith(new UsernameDuplicationValidationStep(userRepository))
.linkWith(new EmailDuplicationValidationStep(userRepository))
.validate(command);
}
private static class CommandConstraintsValidationStep extends ValidationStep<SignUpCommand> {
@Override
public ValidationResult validate(SignUpCommand command) {
try (ValidatorFactory validatorFactory = Validation.buildDefaultValidatorFactory()) {
final Validator validator = validatorFactory.getValidator();
final Set
validate
方法是核心方法,其中调用linkWith
方法组装参数的链式校验器,其中涉及多个验证类,先做基础验证,如果通过的话,去验证用户名是否重复,如果也通过的话,去验证Email
是否重复。CommandConstraintsValidationStep
类,此步骤是一个基础验证,所有的javax validation annotation
都会被验证,比如是否为空,Email
格式是否正确等等。这非常方便,我们不必自己编写这些验证器。如果一个对象是有效的,那么调用checkNext
方法让流程进入下一步,checkNext
,如果不是,ValidationResult
将立即返回。UsernameDuplicationValidationStep
类,此步骤验证用户名是否重复,主要需要去查数据库了。如果是,那么将立即返回无效的ValidationResult
,否则的话继续往后走,去验证下一步。EmailDuplicationValidationStep
类,电子邮件重复验证。因为没有下一步,如果电子邮件是唯一的,则将返回ValidationResult.valid()
。
总结
上面就是通过责任链模式来实现我们参数校验的完整过程了,你学会了吗?这种方式可以优雅的将验证逻辑拆分到单独的类中,如果添加新的验证逻辑,只需要添加新的类,然后组装到“校验链”中。但是在我看来,这比较适合于用于校验相对复杂的场景,如果只是简单的校验就完全没必要这么做了,反而会增加代码的复杂度。
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。
举报投诉
-
数据
+关注
关注
8文章
7006浏览量
88958 -
JAVA
+关注
关注
19文章
2966浏览量
104707 -
代码
+关注
关注
30文章
4780浏览量
68539
发布评论请先 登录
相关推荐
如何用串口进行调试
串口是MCU最重要的一个通信端口,几乎所有的嵌入式产品都会用到串口,可能是产品预研的时候用来调试,可能是与某一个外设连接收发数据,还可能用来对产品进行固件升级。这篇文章主要给大家介绍一下如何用串口进行调试,也就是打印log。
如何用labview进行CRC和LRC校验
最近做的一个课题上用到MODBUS的ASCII模式或RTU模式,分别需要进行LRC和CRC校验,接收方应该怎样校验呢? 查了查资料 LRC的检验(接收方)步骤: 步骤1:对除开始的冒号及结束的回车
发表于 05-11 16:13
曼彻斯特编码解码+CRC校验,进行高速LVDS传输。。代码
曼彻斯特编码解码+CRC校验,进行高速LVDS传输。。代码分两部分。。。第一部分为曼彻斯特编码,编码位数和同步头可以参数化设计,方便移植,数据后面紧接着8为校验码。。。第二部分为曼彻斯
发表于 07-17 22:20
如何利用区块链进行存在性证明?
如果了解区块链原理后,你可以很轻松的理解如何用区块链进行存在性证明,上图VB手拿最新以太坊区块链高度和地址,再配以他的图片很好的证明了他于区
发表于 09-22 09:00
•1480次阅读
如何用SMART编写CRC校验算法程序
CRC即循环冗余校验码(Cyclic Redundancy Check):是数据通信领域中最常用的一种查错校验码,其特征是信息字段和校验字段的长度可以任意选定。
嵌入式C语言可以带“默认参数”的函数吗
使用C++开发过程序时,定义函数可以指定默认参数,例如 void fun(int x, int y=3); 在调用 fun() 时第二个参数可以不传递,此时 fun() 函数默认第二个
发表于 09-06 11:35
•1420次阅读
什么是责任链?
责任链模式是行为模式的一种,它将需要触发的Handler组成一条链,发送者将请求发给链的第一个接收者,并且沿着这条链传递,直到有一个Hand
SpringBoot Web应用如何进行参数校验?(上)
的话就太繁琐了,代码可读性极差。**Validator框架**就是为了解决开发人员在开发的时候少写代码,提升开发效率;Validator专门用来进行接口参数校验,例如常见的必填校验,e
SpringBoot Web应用如何进行参数校验?(下)
代码对接口参数一个个校验的话就太繁琐了,代码可读性极差。Validator框架就是为了解决开发人员在开发的时候少写代码,提升开发效率;Validator专门用来进行接口参数
什么是奇校验和偶校验?常见的奇偶校验方式有哪些?
什么是奇校验和偶校验?常见的奇偶校验方式有哪些? 1. 奇偶校验是指在数字通信中采用一种威廉希尔官方网站
对传输的数据进行
评论