前言
項目中參數(shù)校驗十分重要,它可以保護我們應(yīng)用程序的安全性和合法性。我想大家通常的做法是像下面這樣做的:
@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
}
這么做最大的優(yōu)勢就是簡單直接,但是如果驗證邏輯很復(fù)雜,會導(dǎo)致這個類變得很龐大,而且上面是通過拋出異常來改變代碼執(zhí)行流程,這也是一種不推薦的做法。
那么有什么更好的參數(shù)校驗的方式呢?本文就推薦一種通過責(zé)任鏈設(shè)計模式來優(yōu)雅地實現(xiàn)參數(shù)的校驗功能,我們通過一個用戶注冊的例子來講明白如何實現(xiàn)。
- 有效的注冊數(shù)據(jù)——名字、姓氏、電子郵件、用戶名和密碼。
- 用戶名必須是唯一的。
- 電子郵件必須是唯一的。
定義用戶注冊和驗證結(jié)果類
- 定義一個
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,因為我希望命令對象是不可變的。注冊用戶的數(shù)據(jù)應(yīng)與注冊表中填寫的數(shù)據(jù)相同。
- 定義存儲驗證結(jié)果類
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;
}
}
- 在我看來,這是一種非常方便的方法返回類型,并且比拋出帶有驗證消息的異常要好。
- 既然是責(zé)任鏈,還需要定義一個“鏈”類
ValidationStep,它是這些驗證步驟的超類,我們希望將它們相互“鏈接”起來。
public abstract class ValidationStep
核心驗證邏輯
現(xiàn)在我們開始進行參數(shù)校驗的核心邏輯,也就是如何把上面定義的類給串聯(lián)起來。
- 我們定義一個用于注冊驗證的接口類
SignUpValidationService
public interface SignUpValidationService {
ValidationResult validate(SignUpCommand command);
}
- 現(xiàn)在我們可以使用上面定義的類和責(zé)任鏈模式來輕松的實現(xiàn),代碼如下:
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方法是核心方法,其中調(diào)用linkWith方法組裝參數(shù)的鏈?zhǔn)叫r炂?,其中涉及多個驗證類,先做基礎(chǔ)驗證,如果通過的話,去驗證用戶名是否重復(fù),如果也通過的話,去驗證Email是否重復(fù)。CommandConstraintsValidationStep類,此步驟是一個基礎(chǔ)驗證,所有的javax validation annotation都會被驗證,比如是否為空,Email格式是否正確等等。這非常方便,我們不必自己編寫這些驗證器。如果一個對象是有效的,那么調(diào)用checkNext方法讓流程進入下一步,checkNext,如果不是,ValidationResult將立即返回。UsernameDuplicationValidationStep類,此步驟驗證用戶名是否重復(fù),主要需要去查數(shù)據(jù)庫了。如果是,那么將立即返回?zé)o效的ValidationResult,否則的話繼續(xù)往后走,去驗證下一步。EmailDuplicationValidationStep類,電子郵件重復(fù)驗證。因為沒有下一步,如果電子郵件是唯一的,則將返回ValidationResult.valid()。
總結(jié)
上面就是通過責(zé)任鏈模式來實現(xiàn)我們參數(shù)校驗的完整過程了,你學(xué)會了嗎?這種方式可以優(yōu)雅的將驗證邏輯拆分到單獨的類中,如果添加新的驗證邏輯,只需要添加新的類,然后組裝到“校驗鏈”中。但是在我看來,這比較適合于用于校驗相對復(fù)雜的場景,如果只是簡單的校驗就完全沒必要這么做了,反而會增加代碼的復(fù)雜度。
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。
舉報投訴
-
數(shù)據(jù)
+關(guān)注
關(guān)注
8文章
7335瀏覽量
94761 -
JAVA
+關(guān)注
關(guān)注
20文章
3001瀏覽量
116426 -
代碼
+關(guān)注
關(guān)注
30文章
4968瀏覽量
73960
發(fā)布評論請先 登錄
相關(guān)推薦
熱點推薦
如何用串口進行調(diào)試
串口是MCU最重要的一個通信端口,幾乎所有的嵌入式產(chǎn)品都會用到串口,可能是產(chǎn)品預(yù)研的時候用來調(diào)試,可能是與某一個外設(shè)連接收發(fā)數(shù)據(jù),還可能用來對產(chǎn)品進行固件升級。這篇文章主要給大家介紹一下如何用串口進行調(diào)試,也就是打印log。
如何用labview進行CRC和LRC校驗
最近做的一個課題上用到MODBUS的ASCII模式或RTU模式,分別需要進行LRC和CRC校驗,接收方應(yīng)該怎樣校驗呢? 查了查資料 LRC的檢驗(接收方)步驟: 步驟1:對除開始的冒號及結(jié)束的回車
發(fā)表于 05-11 16:13
曼徹斯特編碼解碼+CRC校驗,進行高速LVDS傳輸。。代碼
曼徹斯特編碼解碼+CRC校驗,進行高速LVDS傳輸。。代碼分兩部分。。。第一部分為曼徹斯特編碼,編碼位數(shù)和同步頭可以參數(shù)化設(shè)計,方便移植,數(shù)據(jù)后面緊接著8為校驗碼。。。第二部分為曼徹斯
發(fā)表于 07-17 22:20
Matlab是如何用系統(tǒng)辨識并進行PID參數(shù)調(diào)節(jié)呢
Matlab是如何用系統(tǒng)辨識并進行PID參數(shù)調(diào)節(jié)呢?有哪些基本步驟呢?
發(fā)表于 11-19 06:27
如何利用區(qū)塊鏈進行存在性證明?
如果了解區(qū)塊鏈原理后,你可以很輕松的理解如何用區(qū)塊鏈進行存在性證明,上圖VB手拿最新以太坊區(qū)塊鏈高度和地址,再配以他的圖片很好的證明了他于區(qū)
發(fā)表于 09-22 09:00
?1960次閱讀
如何用SMART編寫CRC校驗算法程序
CRC即循環(huán)冗余校驗碼(Cyclic Redundancy Check):是數(shù)據(jù)通信領(lǐng)域中最常用的一種查錯校驗碼,其特征是信息字段和校驗字段的長度可以任意選定。
嵌入式C語言可以帶“默認參數(shù)”的函數(shù)嗎
使用C++開發(fā)過程序時,定義函數(shù)可以指定默認參數(shù),例如 void fun(int x, int y=3); 在調(diào)用 fun() 時第二個參數(shù)可以不傳遞,此時 fun() 函數(shù)默認第二個
發(fā)表于 09-06 11:35
?1710次閱讀
如何用MATLAB進行電路仿真
本文演示如何用MATLAB進行電路仿真,測量RLC電路的電壓。我用的是R2014a,不同版本軟件界面稍有差別。
什么是責(zé)任鏈?
責(zé)任鏈模式是行為模式的一種,它將需要觸發(fā)的Handler組成一條鏈,發(fā)送者將請求發(fā)給鏈的第一個接收者,并且沿著這條鏈傳遞,直到有一個Hand
SpringBoot Web應(yīng)用如何進行參數(shù)校驗?(上)
的話就太繁瑣了,代碼可讀性極差。**Validator框架**就是為了解決開發(fā)人員在開發(fā)的時候少寫代碼,提升開發(fā)效率;Validator專門用來進行接口參數(shù)校驗,例如常見的必填校驗,e
SpringBoot Web應(yīng)用如何進行參數(shù)校驗?(下)
代碼對接口參數(shù)一個個校驗的話就太繁瑣了,代碼可讀性極差。Validator框架就是為了解決開發(fā)人員在開發(fā)的時候少寫代碼,提升開發(fā)效率;Validator專門用來進行接口參數(shù)
什么是奇校驗和偶校驗?常見的奇偶校驗方式有哪些?
什么是奇校驗和偶校驗?常見的奇偶校驗方式有哪些? 1. 奇偶校驗是指在數(shù)字通信中采用一種技術(shù)對傳輸?shù)臄?shù)據(jù)進行
如何用責(zé)任鏈默認優(yōu)雅地進行參數(shù)校驗
評論