衡阳派盒市场营销有限公司

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

如何用責(zé)任鏈默認(rèn)優(yōu)雅地進(jìn)行參數(shù)校驗(yàn)

jf_78858299 ? 來源:JAVA旭陽 ? 作者:JAVA旭陽 ? 2023-04-06 15:00 ? 次閱讀

前言

項(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è)類變得很龐大,而且上面是通過拋出異常來改變代碼執(zhí)行流程,這也是一種不推薦的做法。

那么有什么更好的參數(shù)校驗(yàn)的方式呢?本文就推薦一種通過責(zé)任鏈設(shè)計(jì)模式來優(yōu)雅地實(shí)現(xiàn)參數(shù)的校驗(yàn)功能,我們通過一個(gè)用戶注冊(cè)的例子來講明白如何實(shí)現(xiàn)。

  • 有效的注冊(cè)數(shù)據(jù)——名字、姓氏、電子郵件、用戶名和密碼。
  • 用戶名必須是唯一的。
  • 電子郵件必須是唯一的。

定義用戶注冊(cè)和驗(yàn)證結(jié)果類

  1. 定義一個(gè)SignUpCommand類用來接受用戶注冊(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來驗(yàn)證用戶注冊(cè)信息是否有效。
  • 使用lombok的注解@Value,因?yàn)槲蚁M顚?duì)象是不可變的。注冊(cè)用戶的數(shù)據(jù)應(yīng)與注冊(cè)表中填寫的數(shù)據(jù)相同。
  1. 定義存儲(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;
    }
}
  • 在我看來,這是一種非常方便的方法返回類型,并且比拋出帶有驗(yàn)證消息的異常要好。
  1. 既然是責(zé)任鏈,還需要定義一個(gè)“鏈”類ValidationStep,它是這些驗(yàn)證步驟的超類,我們希望將它們相互“鏈接”起來。
public abstract class ValidationStep

核心驗(yàn)證邏輯

現(xiàn)在我們開始進(jìn)行參數(shù)校驗(yàn)的核心邏輯,也就是如何把上面定義的類給串聯(lián)起來。

  1. 我們定義一個(gè)用于注冊(cè)驗(yàn)證的接口SignUpValidationService
public interface SignUpValidationService {
    ValidationResult validate(SignUpCommand command);
}
  1. 現(xiàn)在我們可以使用上面定義的類和責(zé)任鏈模式來輕松的實(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
  • validate方法是核心方法,其中調(diào)用linkWith方法組裝參數(shù)的鏈?zhǔn)叫r?yàn)器,其中涉及多個(gè)驗(yàn)證類,先做基礎(chǔ)驗(yàn)證,如果通過的話,去驗(yàn)證用戶名是否重復(fù),如果也通過的話,去驗(yàn)證Email是否重復(fù)。
  • CommandConstraintsValidationStep類,此步驟是一個(gè)基礎(chǔ)驗(yàn)證,所有的javax validation annotation都會(huì)被驗(yàn)證,比如是否為空,Email格式是否正確等等。這非常方便,我們不必自己編寫這些驗(yàn)證器。如果一個(gè)對(duì)象是有效的,那么調(diào)用checkNext方法讓流程進(jìn)入下一步,checkNext,如果不是,ValidationResult 將立即返回。
  • UsernameDuplicationValidationStep類,此步驟驗(yàn)證用戶名是否重復(fù),主要需要去查數(shù)據(jù)庫了。如果是,那么將立即返回?zé)o效的ValidationResult,否則的話繼續(xù)往后走,去驗(yàn)證下一步。
  • EmailDuplicationValidationStep 類,電子郵件重復(fù)驗(yàn)證。因?yàn)闆]有下一步,如果電子郵件是唯一的,則將返回ValidationResult.valid()

總結(jié)

上面就是通過責(zé)任鏈模式來實(shí)現(xiàn)我們參數(shù)校驗(yàn)的完整過程了,你學(xué)會(huì)了嗎?這種方式可以優(yōu)雅的將驗(yàn)證邏輯拆分到單獨(dú)的類中,如果添加新的驗(yàn)證邏輯,只需要添加新的類,然后組裝到“校驗(yàn)鏈”中。但是在我看來,這比較適合于用于校驗(yàn)相對(duì)復(fù)雜的場(chǎng)景,如果只是簡(jiǎn)單的校驗(yàn)就完全沒必要這么做了,反而會(huì)增加代碼的復(fù)雜度。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 數(shù)據(jù)
    +關(guān)注

    關(guān)注

    8

    文章

    7145

    瀏覽量

    89581
  • JAVA
    +關(guān)注

    關(guān)注

    19

    文章

    2975

    瀏覽量

    105149
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4827

    瀏覽量

    69055
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    何用串口進(jìn)行調(diào)試

    串口是MCU最重要的一個(gè)通信端口,幾乎所有的嵌入式產(chǎn)品都會(huì)用到串口,可能是產(chǎn)品預(yù)研的時(shí)候用來調(diào)試,可能是與某一個(gè)外設(shè)連接收發(fā)數(shù)據(jù),還可能用來對(duì)產(chǎn)品進(jìn)行固件升級(jí)。這篇文章主要給大家介紹一下如何用串口進(jìn)行調(diào)試,也就是打印log。
    的頭像 發(fā)表于 03-23 10:02 ?4663次閱讀
    如<b class='flag-5'>何用</b>串口<b class='flag-5'>進(jìn)行</b>調(diào)試

    何用labview進(jìn)行CRC和LRC校驗(yàn)

    最近做的一個(gè)課題上用到MODBUS的ASCII模式或RTU模式,分別需要進(jìn)行LRC和CRC校驗(yàn),接收方應(yīng)該怎樣校驗(yàn)呢? 查了查資料 LRC的檢驗(yàn)(接收方)步驟: 步驟1:對(duì)除開始的冒號(hào)及結(jié)束的回車
    發(fā)表于 05-11 16:13

    曼徹斯特編碼解碼+CRC校驗(yàn)進(jìn)行高速LVDS傳輸。。代碼

    曼徹斯特編碼解碼+CRC校驗(yàn)進(jìn)行高速LVDS傳輸。。代碼分兩部分。。。第一部分為曼徹斯特編碼,編碼位數(shù)和同步頭可以參數(shù)化設(shè)計(jì),方便移植,數(shù)據(jù)后面緊接著8為校驗(yàn)碼。。。第二部分為曼徹斯
    發(fā)表于 07-17 22:20

    Matlab是如何用系統(tǒng)辨識(shí)并進(jìn)行PID參數(shù)調(diào)節(jié)呢

    Matlab是如何用系統(tǒng)辨識(shí)并進(jìn)行PID參數(shù)調(diào)節(jié)呢?有哪些基本步驟呢?
    發(fā)表于 11-19 06:27

    如何利用區(qū)塊進(jìn)行存在性證明?

    如果了解區(qū)塊原理后,你可以很輕松的理解如何用區(qū)塊進(jìn)行存在性證明,上圖VB手拿最新以太坊區(qū)塊高度和地址,再配以他的圖片很好的證明了他于區(qū)
    發(fā)表于 09-22 09:00 ?1510次閱讀

    自制繼電器電壓電流參數(shù)校驗(yàn)

    筆者研制出一種針對(duì)繼電保護(hù)用的電壓,電流繼電器整定值校驗(yàn)測(cè)試儀。便于技術(shù)人員對(duì)繼電器的整定值進(jìn)行校驗(yàn)和調(diào)整。
    的頭像 發(fā)表于 01-26 08:18 ?3128次閱讀
    自制繼電器電壓電流<b class='flag-5'>參數(shù)</b><b class='flag-5'>校驗(yàn)</b>儀

    何用SMART編寫CRC校驗(yàn)算法程序

    CRC即循環(huán)冗余校驗(yàn)碼(Cyclic Redundancy Check):是數(shù)據(jù)通信領(lǐng)域中最常用的一種查錯(cuò)校驗(yàn)碼,其特征是信息字段和校驗(yàn)字段的長度可以任意選定。
    的頭像 發(fā)表于 11-26 09:50 ?9381次閱讀
    如<b class='flag-5'>何用</b>SMART編寫CRC<b class='flag-5'>校驗(yàn)</b>算法程序

    嵌入式C語言可以帶“默認(rèn)參數(shù)”的函數(shù)嗎

    使用C++開發(fā)過程序時(shí),定義函數(shù)可以指定默認(rèn)參數(shù),例如 void fun(int x, int y=3); 在調(diào)用 fun() 時(shí)第二個(gè)參數(shù)可以不傳遞,此時(shí) fun() 函數(shù)默認(rèn)第二個(gè)
    發(fā)表于 09-06 11:35 ?1449次閱讀
    嵌入式C語言可以帶“<b class='flag-5'>默認(rèn)</b><b class='flag-5'>參數(shù)</b>”的函數(shù)嗎

    何用MATLAB進(jìn)行電路仿真

    本文演示如何用MATLAB進(jìn)行電路仿真,測(cè)量RLC電路的電壓。我用的是R2014a,不同版本軟件界面稍有差別。
    的頭像 發(fā)表于 08-09 17:14 ?1.4w次閱讀

    什么是責(zé)任

    責(zé)任模式是行為模式的一種,它將需要觸發(fā)的Handler組成一條,發(fā)送者將請(qǐng)求發(fā)給的第一個(gè)接收者,并且沿著這條傳遞,直到有一個(gè)Hand
    的頭像 發(fā)表于 02-16 14:41 ?1012次閱讀

    SpringBoot Web應(yīng)用如何進(jìn)行參數(shù)校驗(yàn)?(上)

    的話就太繁瑣了,代碼可讀性極差。**Validator框架**就是為了解決開發(fā)人員在開發(fā)的時(shí)候少寫代碼,提升開發(fā)效率;Validator專門用來進(jìn)行接口參數(shù)校驗(yàn),例如常見的必填校驗(yàn),e
    的頭像 發(fā)表于 05-11 10:31 ?646次閱讀

    SpringBoot Web應(yīng)用如何進(jìn)行參數(shù)校驗(yàn)?(下)

    代碼對(duì)接口參數(shù)一個(gè)個(gè)校驗(yàn)的話就太繁瑣了,代碼可讀性極差。Validator框架就是為了解決開發(fā)人員在開發(fā)的時(shí)候少寫代碼,提升開發(fā)效率;Validator專門用來進(jìn)行接口參數(shù)
    的頭像 發(fā)表于 05-11 10:37 ?567次閱讀

    責(zé)任設(shè)計(jì)模式詳解

    責(zé)任模式是一種行為設(shè)計(jì)模式, 允許你將請(qǐng)求沿著處理者進(jìn)行發(fā)送。收到請(qǐng)求后, 每個(gè)處理者均可對(duì)請(qǐng)求進(jìn)行處理, 或?qū)⑵鋫鬟f給
    的頭像 發(fā)表于 05-22 15:12 ?467次閱讀
    <b class='flag-5'>責(zé)任</b><b class='flag-5'>鏈</b>設(shè)計(jì)模式詳解

    何用MATLAB進(jìn)行電路仿真?

    本文演示如何用MATLAB進(jìn)行電路仿真,測(cè)量RLC電路的電壓。我用的是R2014a,不同版本軟件界面稍有差別。
    的頭像 發(fā)表于 05-26 09:47 ?5172次閱讀
    如<b class='flag-5'>何用</b>MATLAB<b class='flag-5'>進(jìn)行</b>電路仿真?

    什么是奇校驗(yàn)和偶校驗(yàn)?常見的奇偶校驗(yàn)方式有哪些?

    什么是奇校驗(yàn)和偶校驗(yàn)?常見的奇偶校驗(yàn)方式有哪些? 1. 奇偶校驗(yàn)是指在數(shù)字通信中采用一種技術(shù)對(duì)傳輸?shù)臄?shù)據(jù)進(jìn)行
    的頭像 發(fā)表于 10-17 16:28 ?1.1w次閱讀
    百家乐77s| 百家乐官网游戏打水| 尊龙百家乐娱乐网| 百家乐官网招商用语| 百家乐最好打法与投注| 定边县| 百家乐连长| 百家乐官网如何投注技巧| 上海百家乐的玩法技巧和规则| 百家乐官网玩法皇冠现金网| 百家乐园云鼎赌场娱乐网规则| 北京百家乐官网网上投注| 利博百家乐的玩法技巧和规则 | 棋牌娱乐网,| 玩百家乐技巧看路| 百家乐官网庄家抽水| 万宝路百家乐的玩法技巧和规则| 皇冠百家乐官网赢钱皇冠| 大发888线上| 大桥下做生意风水好吗| 大发888下注| 澳门百家乐海星王| 百家乐官网只打一种牌型| 申请百家乐官网会员送彩金| 网络棋牌室| 好用百家乐软件| 开平市| 网上玩百家乐游戏有人挣到钱了吗 | 免费百家乐平预测软件| 旧金山百家乐官网的玩法技巧和规则| 道真| 誉博百家乐327589| 百家乐官网5式直缆打法| 大发888娱乐城亚付宝| 长城百家乐游戏| 金公主百家乐官网现金网| 大发888老虎机网页版| 百家乐平台出租家乐平台出租| 百家乐官网赌场方法| 礼泉县| 大发888下载安装包|