吴忠躺衫网络科技有限公司

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

還在自己實現責任鏈?我建議你造輪子之前先看看這個開源項目

京東云 ? 來源:jf_75140285 ? 作者:jf_75140285 ? 2024-09-20 14:38 ? 次閱讀

1. 前言

設計模式在軟件開發中被廣泛使用。通過使用設計模式,開發人員可以更加高效地開發出高質量的軟件系統,提高代碼的可讀性、可維護性和可擴展性。

責任鏈模式是一種常用的行為型設計模式,它將請求沿著處理鏈進行發送,直到其中一個處理者對請求進行處理為止。在責任鏈模式中,通常會有多個處理者,每個處理者都有一個處理請求的方法。當一個請求到達處理鏈的起點時,會依次傳遞給每個處理者進行處理,直到某個處理者能夠處理該請求。這樣可以保證每個請求都能被處理,并且可以根據實際情況動態地添加或刪除處理者,以滿足不同的需求。

責任鏈模式可以幫助降低系統的耦合度,增加系統的靈活性和可擴展性,其在SpringMVC、Netty等許多框架中均有實現。責任鏈模式常用于以下場景:處理復雜的請求邏輯,例如權限驗證、日志記錄等;避免請求發送者和接收者之間的耦合關系;動態地組織處理流程,以適應不同的請求類型和復雜度。

我們在日常開發中如果要使用責任鏈模式,通常需要自己來實現,但自己臨時實現的責任鏈既不通用,也很容易產生框架與業務代碼耦合不清等問題,增加Code Review 的成本。

Netty的代碼向來以優雅著稱,早年我在閱讀Netty的源碼時,萌生出將其責任鏈的實現應用到業務開發中的想法,之后花了點時間將Netty中責任鏈的實現代碼抽取出來,形成了本項目,也就是pie。pie的核心代碼均來自Netty,絕大部分的 API 與 Netty 是一致的。

pie 是一個可快速上手的責任鏈框架,開發者只需要專注業務,開發相應的業務Handler,即可完成業務的責任鏈落地。

一分鐘學會、三分鐘上手、五分鐘應用,歡迎 star。

pie 源碼地址:https://github.com/feiniaojin/pie.git

pie 案例工程源碼地址:https://github.com/feiniaojin/pie-example.git

2. 快速入門

2.1 引入 maven 依賴

pie 目前已打包發布到 maven 中央倉庫,開發者可以直接通過 maven 坐標將其引入到項目中。


    com.feiniaojin.ddd.ecosystem
    pie
    1.0

目前最新的版本是 1.0

2.2 實現出參工廠

出參也就是執行結果,一般的執行過程都要求有執行結果返回。實現 OutboundFactory 接口,用于產生接口默認返回值。

例如:

public class OutFactoryImpl implements OutboundFactory {
    @Override
    public Object newInstance() {
        Result result = new Result();
        result.setCode(0);
        result.setMsg("ok");
        return result;
    }
}

2.3 實現 handler 接口完成業務邏輯

在 pie 案例工程( https://github.com/feiniaojin/pie-example.git )的 Example1 中,為了展示 pie 的使用方法,實現了一個虛擬的業務邏輯:CMS類項目修改文章標題、正文,大家不要關注修改操作放到兩個 handler 中是否合理,僅作為講解案例。

三個 Handler 功能如下:

CheckParameterHandler:用于參數校驗。

ArticleModifyTitleHandler:用于修改文章的標題。

ArticleModifyContentHandler:用于修改文章的正文。

CheckParameterHandler 的代碼如下:

public class CheckParameterHandler implements ChannelHandler {

    private Logger logger = LoggerFactory.getLogger(CheckParameterHandler.class);

    @Override
    public void channelProcess(ChannelHandlerContext ctx,
                               Object in,
                               Object out) throws Exception {

        logger.info("參數校驗:開始執行");

        if (in instanceof ArticleTitleModifyCmd) {
            ArticleTitleModifyCmd cmd = (ArticleTitleModifyCmd) in;
            String articleId = cmd.getArticleId();
            Objects.requireNonNull(articleId, "articleId不能為空");
            String title = cmd.getTitle();
            Objects.requireNonNull(title, "title不能為空");
            String content = cmd.getContent();
            Objects.requireNonNull(content, "content不能為空");
        }
        logger.info("參數校驗:校驗通過,即將進入下一個Handler");
        ctx.fireChannelProcess(in, out);
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx,
                                Throwable cause,
                                Object in,
                                Object out) throws Exception {
        logger.error("參數校驗:異常處理邏輯", cause);
        Result re = (Result) out;
        re.setCode(400);
        re.setMsg("參數異常");
    }
}

ArticleModifyTitleHandler 的代碼如下:

public class ArticleModifyTitleHandler implements ChannelHandler {

    private Logger logger = LoggerFactory.getLogger(ArticleModifyTitleHandler.class);

    @Override
    public void channelProcess(ChannelHandlerContext ctx,
                               Object in,
                               Object out) throws Exception {

        logger.info("修改標題:進入修改標題的Handler");

        ArticleTitleModifyCmd cmd = (ArticleTitleModifyCmd) in;

        String title = cmd.getTitle();
        //修改標題的業務邏輯
        logger.info("修改標題:title={}", title);

        logger.info("修改標題:執行完成,即將進入下一個Handler");
        ctx.fireChannelProcess(in, out);
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx,
                                Throwable cause,
                                Object in,
                                Object out) throws Exception {
        logger.error("修改標題:異常處理邏輯");
        Result re = (Result) out;
        re.setCode(1501);
        re.setMsg("修改標題發生異常");
    }
}

ArticleModifyContentHandler 的代碼如下:

public class ArticleModifyContentHandler implements ChannelHandler {

    private Logger logger = LoggerFactory.getLogger(ArticleModifyContentHandler.class);

    @Override
    public void channelProcess(ChannelHandlerContext ctx,
                               Object in,
                               Object out) throws Exception {

        logger.info("修改正文:進入修改正文的Handler");
        ArticleTitleModifyCmd cmd = (ArticleTitleModifyCmd) in;
        logger.info("修改正文,content={}", cmd.getContent());
        logger.info("修改正文:執行完成,即將進入下一個Handler");
        ctx.fireChannelProcess(in, out);
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx,
                                Throwable cause,
                                Object in,
                                Object out) throws Exception {

        logger.error("修改標題:異常處理邏輯");

        Result re = (Result) out;
        re.setCode(1502);
        re.setMsg("修改正文發生異常");
    }
}

2.4 通過 BootStrap 拼裝并執行

public class ArticleModifyExample1 {

    private final static Logger logger = LoggerFactory.getLogger(ArticleModifyExample1.class);

    public static void main(String[] args) {
        //構造入參
        ArticleTitleModifyCmd dto = new ArticleTitleModifyCmd();
        dto.setArticleId("articleId_001");
        dto.setTitle("articleId_001_title");
        dto.setContent("articleId_001_content");

        //創建引導類
        BootStrap bootStrap = new BootStrap();

        //拼裝并執行
        Result result = (Result) bootStrap
                .inboundParameter(dto)//入參
                .outboundFactory(new ResultFactory())//出參工廠
                .channel(new ArticleModifyChannel())//自定義channel
                .addChannelHandlerAtLast("checkParameter", new CheckParameterHandler())//第一個handler
                .addChannelHandlerAtLast("modifyTitle", new ArticleModifyTitleHandler())//第二個handler
                .addChannelHandlerAtLast("modifyContent", new ArticleModifyContentHandler())//第三個handler
                .process();//執行
        //result為執行結果
        logger.info("result:code={},msg={}", result.getCode(), result.getMsg());
    }
}

2.5 執行結果

以下是運行 ArticleModifyExample1 的 main 方法打出的日志,可以看到我們定義的 handler 被逐個執行了。

wKgaombtGFuAFlFjAAMfA3tGG9c190.png

3. 異常處理

3.1 Handler 異常處理

當某個Handler執行發生異常時,我們可將其異常處理邏輯實現在當前 Handler 的 exceptionCaught 方法中。

在 pie 案例工程( https://github.com/feiniaojin/pie-example.git )的 example2 包中,展示了某個 Handler 拋出異常時的處理方式。

假設 ArticleModifyTitleHandler 的業務邏輯會拋出異常,實例代碼如下:

public class ArticleModifyTitleHandler implements ChannelHandler {

    private Logger logger = LoggerFactory.getLogger(ArticleModifyTitleHandler.class);

    @Override
    public void channelProcess(ChannelHandlerContext ctx,
                               Object in,
                               Object out) throws Exception {

        logger.info("修改標題:進入修改標題的Handler");
        ArticleTitleModifyCmd cmd = (ArticleTitleModifyCmd) in;
        String title = cmd.getTitle();
        //此處的異常用于模擬執行過程中出現異常的場景
        throw new RuntimeException("修改title發生異常");
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx,
                                Throwable cause,
                                Object in,
                                Object out) throws Exception {
        logger.error("修改標題:異常處理邏輯");
        Result re = (Result) out;
        re.setCode(1501);
        re.setMsg("修改標題發生異常");
    }
}

此時 ArticleModifyTitleHandler 的 channelProcess 方法一定會拋出異常, 在當前 Handler 的 exceptionCaught 方法中對異常進行了處理。

運行 ArticleModifyExample2 的 main 方法,輸出如下:

wKgaombtGGeAdo9GAAHqAaCzzZs111.png

3.2 全局異常處理

有時候,我們不想每個 handler 都處理一遍異常,我們希望在執行鏈的最后統一進行處理。
在 ArticleModifyExample3 中,我們展示了通過一個全局異常進行最后的異常處理,其實現主要分為以下幾步:

3.2.1 業務 Handler 傳遞異常

如果業務 Handler 實現了 ChannelHandler 接口,那么需要手工調用 ctx.fireExceptionCaught 方法向下傳遞異常。
例如 CheckParameterHandler 捕獲到異常時的示例如下:


@Override
public class XXXHandler implements ChannelHandler {

    //省略其他邏輯

    //異常處理
    public void exceptionCaught(ChannelHandlerContext ctx,
                                Throwable cause,
                                Object in,
                                Object out) throws Exception {

        logger.info("參數校驗的異常處理邏輯:不處理直接向后傳遞");
        ctx.fireExceptionCaught(cause, in, out);
    }
}

如果業務 Handler 繼承了 ChannelHandlerAdapter,如果沒有重寫 fireExceptionCaught 方法,則默認將異常向后傳遞。

3.2.2 實現全局異常處理的 Handler

我們把業務異常處理邏輯放到最后的 Handler 中進行處理,該 Handler 繼承了ChannelHandlerAdapter,只需要重寫異常處理的exceptionCaught
方法。
示例代碼如下:

public class ExceptionHandler extends ChannelHandlerAdapter {

    private Logger logger = LoggerFactory.getLogger(ExceptionHandler.class);

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx,
                                Throwable cause,
                                Object in,
                                Object out) throws Exception {

        logger.error("異常處理器中的異常處理邏輯");
        Result re = (Result) out;
        re.setCode(500);
        re.setMsg("系統異常");
    }
}

3.2.3 將 ExceptionHandler 加入到執行鏈中

直接通過 BootStrap 加入到執行鏈最后即可,示例代碼如下:


public class ArticleModifyExample3 {

    private final static Logger logger = LoggerFactory.getLogger(ArticleModifyExample3.class);

    public static void main(String[] args) {
        //入參
        ArticleTitleModifyCmd dto = new ArticleTitleModifyCmd();
        dto.setArticleId("articleId_001");
        dto.setTitle("articleId_001_title");
        dto.setContent("articleId_001_content");
        //創建引導類
        BootStrap bootStrap = new BootStrap();

        Result result = (Result) bootStrap
                .inboundParameter(dto)//入參
                .outboundFactory(new ResultFactory())//出參工廠
                .channel(new ArticleModifyChannel())//自定義channel
                .addChannelHandlerAtLast("checkParameter", new CheckParameterHandler())//第一個handler
                .addChannelHandlerAtLast("modifyTitle", new ArticleModifyTitleHandler())//第二個handler
                .addChannelHandlerAtLast("modifyContent", new ArticleModifyContentHandler())//第三個handler
                .addChannelHandlerAtLast("exception", new ExceptionHandler())//異常處理handler
                .process();//執行
        //result為執行結果
        logger.info("result:code={},msg={}", result.getCode(), result.getMsg());
    }
}

3.2.4 運行 ArticleModifyExample3

運行 ArticleModifyExample3 的 main 方法,控制臺輸出如下,可以看到異常被傳遞到最后的 ExceptionHandler 中進行處理。

wKgZombtGGmAT1MbAAKoayl5eCE437.png

4. 總結

本文通過簡單的例子,向讀者介紹了如何使用pie框架快速進行責任鏈模式開發,包括責任鏈初始化和異常處理等日常開發中常見的場景。讀者可以參考這些案例,并將pie框架應用于實際開發中,以快速實現通用的責任鏈模式,最終降低代碼的耦合度、增加代碼的可擴展性和提高代碼的可讀性。

審核編輯 黃宇

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 接口
    +關注

    關注

    33

    文章

    8694

    瀏覽量

    151928
  • 開源
    +關注

    關注

    3

    文章

    3408

    瀏覽量

    42714
  • 源碼
    +關注

    關注

    8

    文章

    652

    瀏覽量

    29458
收藏 人收藏

    評論

    相關推薦

    ElfBoard開源項目|智能消防車項目

    項目——利用ELF 1開發板打造的智慧消防車。該項目展現了物聯網、數據處理等前沿技術在消防領域的應用潛力,下面就和各位小伙伴展示一下這個開源項目
    的頭像 發表于 01-04 16:43 ?324次閱讀
    ElfBoard<b class='flag-5'>開源</b><b class='flag-5'>項目</b>|智能消防車<b class='flag-5'>項目</b>

    開源鴻蒙榮獲開放原子“2024年度操作系統領域國內活躍開源項目

    近日,2024開放原子開發者大會暨首屆開源技術學術大會在武漢圓滿召開。在大會開幕式“2024年度國內活躍開源項目&開發者致謝儀式”上,開放原子開源鴻蒙(OpenAtom OpenHar
    的頭像 發表于 12-28 15:39 ?480次閱讀

    開源項目!OpenCat—— 一個全能的平價四足機器人

    的消費市場。可以把它想象成一個有腿的安卓手機或Alexa,帶有一個可供第三方擴展的應用商店。它可以以約每秒 2.6 個身體長度的速度持續運行 60分鐘,或者坐著播放視頻幾個小時。還在脊柱下方預留了一些
    發表于 12-16 11:44

    M5Stack Hackathon:看看我們都做了什么好玩的項目!

    眾所周知(?),M5Stack于2024年11月30日舉行了內部員工的Hackathon大賽,要求用我們自己生產的產品,在一天的時間完成項目制作以及陳述。讓我們一起來看看大家做了什么好玩的項目
    的頭像 發表于 12-07 01:08 ?278次閱讀
    M5Stack Hackathon:<b class='flag-5'>看看</b>我們都做了什么好玩的<b class='flag-5'>項目</b>!

    ElfBoard開源項目|車牌識別項目技術文檔

    車牌識別項目基于百度智能云平臺,旨在利用其強大的OCR服務實現車牌號碼的自動識別。選擇百度智能云的原因是其高效的API接口和穩定的服務質量,能夠幫助開發者快速實現車牌識別應用。這個
    的頭像 發表于 12-06 10:30 ?316次閱讀
    ElfBoard<b class='flag-5'>開源</b><b class='flag-5'>項目</b>|車牌識別<b class='flag-5'>項目</b>技術文檔

    凌蒙派OpenHarmony開源項目榮獲本期Gitee官方推薦

    近日,司凌蒙派OpenHarmony開源項目榮獲本期Gitee官方推薦。本期Gitee官方推薦不僅是對凌蒙派OpenHarmony開源項目
    的頭像 發表于 11-20 01:04 ?434次閱讀
    凌蒙派OpenHarmony<b class='flag-5'>開源</b><b class='flag-5'>項目</b>榮獲本期Gitee官方推薦

    明明說的是25G信號,卻讓看12.5G的損耗?

    ,就三下五除二脫口而出:12.5G是7.3dB。客戶聽完都會有這樣的反應:等等,說的是25G的信號哦,告訴12.5G干嘛? 是的,高速先生還真沒弄錯,一般這個時候我們就會直接拿
    發表于 10-23 09:11

    ElfBoard開源項目|“智慧光伏”開源項目技術文檔

    最大化地捕捉和利用。下面就和各位小伙伴詳細介紹一下這一開源項目是怎樣實現的。環境說明1.開發環境操作系統:Ubuntu18.0464位版2.交叉編譯工具:arm-
    的頭像 發表于 10-11 15:52 ?463次閱讀
    ElfBoard<b class='flag-5'>開源</b><b class='flag-5'>項目</b>|“智慧光伏”<b class='flag-5'>開源</b><b class='flag-5'>項目</b>技術文檔

    Matepad pro12.2 已上市半個月,但是還沒有在開源網站看到該項目開源信息,違背開源精神

    Matepad pro12.2 已上市半個月,本人自己也購買了同款12+256的pad,想要同步學習下這款pad的一些體驗還不錯的功能點,但是目前為止還沒有在開源網站看到該項目開源
    發表于 08-27 17:25

    開源項目!基于 Arduino DIY 漂亮的宏機械鍵盤

    。 接下來,我們將要制作的這款鍵盤擁有 12 個按鍵,可以根據自己的喜好和需求,將這些按鍵與所需的快捷鍵進行關聯。盡管這個項目聽起來可能有些復雜,但實際上無論是電子部分還是軟件部分,
    發表于 08-19 17:02

    開源項目!用ESP8266 DIY會爬墻的無人機

    有點神奇,作者表示,下面介紹的這個無人機就能做到! 一個基于NodeMCU飛行控制器和安卓應用控制的爬墻項目。完整的細節和解釋視頻。 相信,會愛上它。 玩無人機需要一個飛行控制器,
    發表于 07-04 09:22

    華為車究竟成沒成功,這個責任誰來擔?

    華為車,一種很新的車方式。
    的頭像 發表于 06-20 11:16 ?316次閱讀

    嵌入式軟件工程師如何提升自己?

    、分享項目經驗等方式,展示自己的專業能力和行業見解,樹立自己在行業中的聲譽。 6.尋找合適的導師 在職業生涯的發展過程中,有一個經驗豐富的導師是非常寶貴的。他們可以給予指導和
    發表于 06-12 11:20

    飛凌嵌入式-ElfBoard-項目分享|基于ELF 1S開發板完成的物聯網開源項目

    工程師借助ELF 1S開發板完成的嵌入式物聯網項目,見證智慧與技術的火花。 關于這個項目的思維導圖,如下圖所示: 接下來,將會從云-網-邊-端的物聯網體系架構出發,描述整個開源
    發表于 05-16 09:12

    項目分享|基于ELF 1S開發板完成的物聯網開源項目

    與技術的火花。關于這個項目的思維導圖,如下圖所示:接下來,將會從云-網-邊-端的物聯網體系架構出發,描述整個開源項目的開發過程,整個開源
    的頭像 發表于 05-14 09:03 ?1420次閱讀
    <b class='flag-5'>項目</b>分享|基于ELF 1S開發板完成的物聯網<b class='flag-5'>開源</b><b class='flag-5'>項目</b>
    伯爵百家乐的玩法技巧和规则| 百家乐官网全自动分析软件| 24山分金吉凶断| 至尊国际娱乐| 百家乐在线怎么玩| 龙岩市| 网上百家乐怎么赌能赢钱| 菲律宾百家乐官网排行| 免费百家乐缩水软件| 百家乐官网庄闲局部失| CEO百家乐的玩法技巧和规则| 五星百家乐官网的玩法技巧和规则 | 百家乐官网是多少个庄闲| 网页百家乐游戏下载| 百家乐套装| 百家乐官网桌子豪华| 优博平台| 最新全讯网网址| 百家乐平台导航| 鲨鱼百家乐官网游戏平台| 大发888娱乐场菲律宾| 澳门百家乐技巧经| 澳门百家乐官网鸿运| 境外赌博下| 太阳城金旭园| 百家乐北京| 做生意风水| 百家乐官网龙虎| 百家乐官网开户送十元| 加多宝百家乐的玩法技巧和规则 | 百家乐研究| 试玩百家乐的玩法技巧和规则 | 博九网百家乐现金网| 赌百家乐官网2号破解| 岳阳县| 一起pk棋牌游戏下载| 百家乐拍照看| 带有百家乐的棋牌游戏有哪些| 聚龍社百家乐官网的玩法技巧和规则| 磐石市| 大发888合营商|