詳解Java如何使用責(zé)任鏈默認(rèn)優(yōu)雅地進(jìn)行參數(shù)校驗(yàn)
前言
項(xiàng)目中參數(shù)校驗(yàn)十分重要,它可以保護(hù)我們應(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)勢(shì)就是簡(jiǎn)單直接,但是如果驗(yàn)證邏輯很復(fù)雜,會(huì)導(dǎo)致這個(gè)類變得很龐大,而且上面是通過(guò)拋出異常來(lái)改變代碼執(zhí)行流程,這也是一種不推薦的做法。
那么有什么更好的參數(shù)校驗(yàn)的方式呢?本文就推薦一種通過(guò)責(zé)任鏈設(shè)計(jì)模式來(lái)優(yōu)雅地實(shí)現(xiàn)參數(shù)的校驗(yàn)功能,我們通過(guò)一個(gè)用戶注冊(cè)的例子來(lái)講明白如何實(shí)現(xiàn)。
- 有效的注冊(cè)數(shù)據(jù)——名字、姓氏、電子郵件、用戶名和密碼。
- 用戶名必須是唯一的。
- 電子郵件必須是唯一的。
定義用戶注冊(cè)和驗(yàn)證結(jié)果類
1.定義一個(gè)SignUpCommand
類用來(lái)接受用戶注冊(cè)的屬性信息。并且使用 @Value
注解讓這個(gè)類不可變。
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
來(lái)驗(yàn)證用戶注冊(cè)信息是否有效。 - 使用
lombok
的注解@Value
,因?yàn)槲蚁M顚?duì)象是不可變的。注冊(cè)用戶的數(shù)據(jù)應(yīng)與注冊(cè)表中填寫(xiě)的數(shù)據(jù)相同。
2.定義存儲(chǔ)驗(yàn)證結(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; } }
在我看來(lái),這是一種非常方便的方法返回類型,并且比拋出帶有驗(yàn)證消息的異常要好。
3.既然是責(zé)任鏈,還需要定義一個(gè)“鏈”類ValidationStep
,它是這些驗(yàn)證步驟的超類,我們希望將它們相互“鏈接”起來(lái)。
public abstract class ValidationStep<T> { private ValidationStep<T> next; public ValidationStep<T> linkWith(ValidationStep<T> next) { if (this.next == null) { this.next = next; return this; } ValidationStep<T> lastStep = this.next; while (lastStep.next != null) { lastStep = lastStep.next; } lastStep.next = next; return this; } public abstract ValidationResult validate(T toValidate); protected ValidationResult checkNext(T toValidate) { if (next == null) { return ValidationResult.valid(); } return next.validate(toValidate); } }
核心驗(yàn)證邏輯
現(xiàn)在我們開(kāi)始進(jìn)行參數(shù)校驗(yàn)的核心邏輯,也就是如何把上面定義的類給串聯(lián)起來(lái)。
1.我們定義一個(gè)用于注冊(cè)驗(yàn)證的接口類SignUpValidationService
public interface SignUpValidationService { ValidationResult validate(SignUpCommand command); }
2.現(xiàn)在我們可以使用上面定義的類和責(zé)任鏈模式來(lái)輕松的實(shí)現(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<ConstraintViolation<SignUpCommand>> constraintsViolations = validator.validate(command); if (!constraintsViolations.isEmpty()) { return ValidationResult.invalid(constraintsViolations.iterator().next().getMessage()); } } return checkNext(command); } } @AllArgsConstructor private static class UsernameDuplicationValidationStep extends ValidationStep<SignUpCommand> { private final UserRepository userRepository; @Override public ValidationResult validate(SignUpCommand command) { if (userRepository.findByUsername(command.getUsername()).isPresent()) { return ValidationResult.invalid(String.format("Username [%s] is already taken", command.getUsername())); } return checkNext(command); } } @AllArgsConstructor private static class EmailDuplicationValidationStep extends ValidationStep<SignUpCommand> { private final UserRepository userRepository; @Override public ValidationResult validate(SignUpCommand command) { if (userRepository.findByEmail(command.getEmail()).isPresent()) { return ValidationResult.invalid(String.format("Email [%s] is already taken", command.getEmail())); } return checkNext(command); } } }
validate
方法是核心方法,其中調(diào)用linkWith
方法組裝參數(shù)的鏈?zhǔn)叫r?yàn)器,其中涉及多個(gè)驗(yàn)證類,先做基礎(chǔ)驗(yàn)證,如果通過(guò)的話,去驗(yàn)證用戶名是否重復(fù),如果也通過(guò)的話,去驗(yàn)證Email
是否重復(fù)。CommandConstraintsValidationStep
類,此步驟是一個(gè)基礎(chǔ)驗(yàn)證,所有的javax validation annotation
都會(huì)被驗(yàn)證,比如是否為空,Email
格式是否正確等等。這非常方便,我們不必自己編寫(xiě)這些驗(yàn)證器。如果一個(gè)對(duì)象是有效的,那么調(diào)用checkNext
方法讓流程進(jìn)入下一步,checkNext
,如果不是,ValidationResult
將立即返回。UsernameDuplicationValidationStep
類,此步驟驗(yàn)證用戶名是否重復(fù),主要需要去查數(shù)據(jù)庫(kù)了。如果是,那么將立即返回?zé)o效的ValidationResult
,否則的話繼續(xù)往后走,去驗(yàn)證下一步。EmailDuplicationValidationStep
類,電子郵件重復(fù)驗(yàn)證。因?yàn)闆](méi)有下一步,如果電子郵件是唯一的,則將返回ValidationResult.valid()
。
總結(jié)
上面就是通過(guò)責(zé)任鏈模式來(lái)實(shí)現(xiàn)我們參數(shù)校驗(yàn)的完整過(guò)程了,你學(xué)會(huì)了嗎?這種方式可以優(yōu)雅的將驗(yàn)證邏輯拆分到單獨(dú)的類中,如果添加新的驗(yàn)證邏輯,只需要添加新的類,然后組裝到“校驗(yàn)鏈”中。但是在我看來(lái),這比較適合于用于校驗(yàn)相對(duì)復(fù)雜的場(chǎng)景,如果只是簡(jiǎn)單的校驗(yàn)就完全沒(méi)必要這么做了,反而會(huì)增加代碼的復(fù)雜度。
到此這篇關(guān)于詳解Java如何使用責(zé)任鏈默認(rèn)優(yōu)雅地進(jìn)行參數(shù)校驗(yàn)的文章就介紹到這了,更多相關(guān)Java責(zé)任鏈實(shí)現(xiàn)參數(shù)校驗(yàn)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java實(shí)現(xiàn)從Html文本中提取純文本的方法
今天小編就為大家分享一篇Java實(shí)現(xiàn)從Html文本中提取純文本的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-05-05說(shuō)說(shuō)@ModelAttribute在父類和子類中的執(zhí)行順序
這篇文章主要介紹了@ModelAttribute在父類和子類中的執(zhí)行順序,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06Java集合Iterator迭代的實(shí)現(xiàn)方法
這篇文章主要介紹了Java集合Iterator迭代接口的實(shí)現(xiàn)方法,非常不錯(cuò),具有參考借鑒家,對(duì)Java 結(jié)合iterator知識(shí)感興趣的朋友一起看看吧2016-08-08詳解Reactor如何優(yōu)雅Exception異常處理
初識(shí)響應(yīng)式編程的時(shí)候,除了從命令式的思維方式轉(zhuǎn)變?yōu)楹瘮?shù)式的編程方式外,其中有一個(gè)很大的不適應(yīng)的地方就是在面對(duì)異常時(shí)該怎么處理。本文將通過(guò)Project?Reactor的文檔以及源碼來(lái)深入解讀,在reactor中是如何優(yōu)雅地實(shí)現(xiàn)這異常處理三板斧,希望對(duì)大家有所幫助2023-02-02Java多線程面試題之交替輸出問(wèn)題的實(shí)現(xiàn)
本文主要介紹了Java多線程面試題之交替輸出問(wèn)題的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01Java中對(duì)于并發(fā)問(wèn)題的處理思路分享
并發(fā)粗暴的解釋就是一段代碼,在同一時(shí)間段內(nèi),被多個(gè)線程同時(shí)處理的情況就是并發(fā)現(xiàn)象。這篇文章和大家分享了一些對(duì)于并發(fā)問(wèn)題的處理思路,需要的可以參考一下2023-02-02