chinese直男口爆体育生外卖, 99久久er热在这里只有精品99, 又色又爽又黄18禁美女裸身无遮挡, gogogo高清免费观看日本电视,私密按摩师高清版在线,人妻视频毛茸茸,91论坛 兴趣闲谈,欧美 亚洲 精品 8区,国产精品久久久久精品免费

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

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

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

一起看看責(zé)任鏈設(shè)計模式吧!

Android編程精選 ? 來源:CSDN ? 作者:Java小海 ? 2022-07-08 16:25 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

  • 什么是責(zé)任鏈
  • 場景
    • 反例
    • 初步改造
    • 缺點(diǎn)
    • 責(zé)任鏈改造
    • 責(zé)任鏈工廠改造
  • 聊聊其他

最近,我讓團(tuán)隊內(nèi)一位成員寫了一個導(dǎo)入功能。他使用了責(zé)任鏈模式,代碼堆的非常多,bug 也多,沒有達(dá)到我預(yù)期的效果。實際上,針對導(dǎo)入功能,我認(rèn)為模版方法更合適!為此,隔壁團(tuán)隊也拿出我們的案例,進(jìn)行了集體 code review。

學(xué)好設(shè)計模式,且不要為了練習(xí),強(qiáng)行使用!讓原本100行就能實現(xiàn)的功能,寫了 3000 行!

對錯暫且不論,我們先一起看看責(zé)任鏈設(shè)計模式吧!

什么是責(zé)任鏈

「責(zé)任鏈模式」 是一種行為設(shè)計模式, 允許你將請求沿著處理者鏈進(jìn)行發(fā)送。收到請求后, 每個處理者均可對請求進(jìn)行處理, 或?qū)⑵鋫鬟f給鏈上的下個處理者。

3dca85f4-fc56-11ec-ba43-dac502259ad0.png

基于 Spring Boot + MyBatis Plus + Vue & Element 實現(xiàn)的后臺管理系統(tǒng) + 用戶小程序,支持 RBAC 動態(tài)權(quán)限、多租戶、數(shù)據(jù)權(quán)限、工作流、三方登錄、支付、短信、商城等功能。

項目地址:https://github.com/YunaiV/ruoyi-vue-pro

場景

責(zé)任鏈的使用場景還是比較多的

  • 多條件流程判斷:權(quán)限控制
  • ERP 系統(tǒng)流程審批:總經(jīng)理、人事經(jīng)理、項目經(jīng)理
  • Java 過濾器 的底層實現(xiàn) Filter

如果不使用該設(shè)計模式,那么當(dāng)需求有所改變時,就會使得代碼臃腫或者難以維護(hù),例如下面的例子

反例

假設(shè)現(xiàn)在有一個闖關(guān)游戲,進(jìn)入下一關(guān)的條件是上一關(guān)的分?jǐn)?shù)要高于xx

  1. 游戲一共 3 個關(guān)卡
  2. 進(jìn)入第二關(guān)需要第一關(guān)的游戲得分大于等于 80
  3. 進(jìn)入第三關(guān)需要第二關(guān)的游戲得分大于等于 90

那么代碼可以這樣寫

//第一關(guān)
publicclassFirstPassHandler{
publicinthandler(){
System.out.println("第一關(guān)-->FirstPassHandler");
return80;
}
}

//第二關(guān)
publicclassSecondPassHandler{
publicinthandler(){
System.out.println("第二關(guān)-->SecondPassHandler");
return90;
}
}


//第三關(guān)
publicclassThirdPassHandler{
publicinthandler(){
System.out.println("第三關(guān)-->ThirdPassHandler,這是最后一關(guān)啦");
return95;
}
}

//客戶端
publicclassHandlerClient{
publicstaticvoidmain(String[]args){

FirstPassHandlerfirstPassHandler=newFirstPassHandler();//第一關(guān)
SecondPassHandlersecondPassHandler=newSecondPassHandler();//第二關(guān)
ThirdPassHandlerthirdPassHandler=newThirdPassHandler();//第三關(guān)

intfirstScore=firstPassHandler.handler();
//第一關(guān)的分?jǐn)?shù)大于等于80則進(jìn)入第二關(guān)
if(firstScore>=80){
intsecondScore=secondPassHandler.handler();
//第二關(guān)的分?jǐn)?shù)大于等于90則進(jìn)入第二關(guān)
if(secondScore>=90){
thirdPassHandler.handler();
}
}
}
}

那么如果這個游戲有100關(guān),我們的代碼很可能就會寫成這個樣子

if(第1關(guān)通過){
//第2關(guān)游戲
if(第2關(guān)通過){
//第3關(guān)游戲
if(第3關(guān)通過){
//第4關(guān)游戲
if(第4關(guān)通過){
//第5關(guān)游戲
if(第5關(guān)通過){
//第6關(guān)游戲
if(第6關(guān)通過){
//...
}
}
}
}
}
}

這種代碼不僅冗余,并且當(dāng)我們要將某兩關(guān)進(jìn)行調(diào)整時會對代碼非常大的改動,這種操作的風(fēng)險是很高的,因此,該寫法非常糟糕

初步改造

如何解決這個問題,我們可以通過鏈表將每一關(guān)連接起來,形成責(zé)任鏈的方式,第一關(guān)通過后是第二關(guān),第二關(guān)通過后是第三關(guān) ....,這樣客戶端就不需要進(jìn)行多重 if 的判斷了

publicclassFirstPassHandler{
/**
*第一關(guān)的下一關(guān)是第二關(guān)
*/
privateSecondPassHandlersecondPassHandler;

publicvoidsetSecondPassHandler(SecondPassHandlersecondPassHandler){
this.secondPassHandler=secondPassHandler;
}

//本關(guān)卡游戲得分
privateintplay(){
return80;
}

publicinthandler(){
System.out.println("第一關(guān)-->FirstPassHandler");
if(play()>=80){
//分?jǐn)?shù)>=80并且存在下一關(guān)才進(jìn)入下一關(guān)
if(this.secondPassHandler!=null){
returnthis.secondPassHandler.handler();
}
}

return80;
}
}

publicclassSecondPassHandler{

/**
*第二關(guān)的下一關(guān)是第三關(guān)
*/
privateThirdPassHandlerthirdPassHandler;

publicvoidsetThirdPassHandler(ThirdPassHandlerthirdPassHandler){
this.thirdPassHandler=thirdPassHandler;
}

//本關(guān)卡游戲得分
privateintplay(){
return90;
}

publicinthandler(){
System.out.println("第二關(guān)-->SecondPassHandler");

if(play()>=90){
//分?jǐn)?shù)>=90并且存在下一關(guān)才進(jìn)入下一關(guān)
if(this.thirdPassHandler!=null){
returnthis.thirdPassHandler.handler();
}
}

return90;
}
}

publicclassThirdPassHandler{

//本關(guān)卡游戲得分
privateintplay(){
return95;
}

/**
*這是最后一關(guān),因此沒有下一關(guān)
*/
publicinthandler(){
System.out.println("第三關(guān)-->ThirdPassHandler,這是最后一關(guān)啦");
returnplay();
}
}

publicclassHandlerClient{
publicstaticvoidmain(String[]args){

FirstPassHandlerfirstPassHandler=newFirstPassHandler();//第一關(guān)
SecondPassHandlersecondPassHandler=newSecondPassHandler();//第二關(guān)
ThirdPassHandlerthirdPassHandler=newThirdPassHandler();//第三關(guān)

firstPassHandler.setSecondPassHandler(secondPassHandler);//第一關(guān)的下一關(guān)是第二關(guān)
secondPassHandler.setThirdPassHandler(thirdPassHandler);//第二關(guān)的下一關(guān)是第三關(guān)

//說明:因為第三關(guān)是最后一關(guān),因此沒有下一關(guān)
//開始調(diào)用第一關(guān)每一個關(guān)卡是否進(jìn)入下一關(guān)卡在每個關(guān)卡中判斷
firstPassHandler.handler();

}
}

缺點(diǎn)

現(xiàn)有模式的缺點(diǎn)

  • 每個關(guān)卡中都有下一關(guān)的成員變量并且是不一樣的,形成鏈很不方便
  • 代碼的擴(kuò)展性非常不好

責(zé)任鏈改造

  • 既然每個關(guān)卡中都有下一關(guān)的成員變量并且是不一樣的,那么我們可以在關(guān)卡上抽象出一個父類或者接口,然后每個具體的關(guān)卡去繼承或者實現(xiàn)

有了思路,我們先來簡單介紹一下責(zé)任鏈設(shè)計模式的「基本組成」

  • 抽象處理者(Handler)角色:定義一個處理請求的接口,包含抽象處理方法和一個后繼連接。
  • 具體處理者(Concrete Handler)角色:實現(xiàn)抽象處理者的處理方法,判斷能否處理本次請求,如果可以處理請求則處理,否則將該請求轉(zhuǎn)給它的后繼者。
  • 客戶類(Client)角色:創(chuàng)建處理鏈,并向鏈頭的具體處理者對象提交請求,它不關(guān)心處理細(xì)節(jié)和請求的傳遞過程。
3de1a3a6-fc56-11ec-ba43-dac502259ad0.jpg責(zé)任鏈改造
publicabstractclassAbstractHandler{

/**
*下一關(guān)用當(dāng)前抽象類來接收
*/
protectedAbstractHandlernext;

publicvoidsetNext(AbstractHandlernext){
this.next=next;
}

publicabstractinthandler();
}

publicclassFirstPassHandlerextendsAbstractHandler{

privateintplay(){
return80;
}

@Override
publicinthandler(){
System.out.println("第一關(guān)-->FirstPassHandler");
intscore=play();
if(score>=80){
//分?jǐn)?shù)>=80并且存在下一關(guān)才進(jìn)入下一關(guān)
if(this.next!=null){
returnthis.next.handler();
}
}
returnscore;
}
}

publicclassSecondPassHandlerextendsAbstractHandler{

privateintplay(){
return90;
}

publicinthandler(){
System.out.println("第二關(guān)-->SecondPassHandler");

intscore=play();
if(score>=90){
//分?jǐn)?shù)>=90并且存在下一關(guān)才進(jìn)入下一關(guān)
if(this.next!=null){
returnthis.next.handler();
}
}

returnscore;
}
}

publicclassThirdPassHandlerextendsAbstractHandler{

privateintplay(){
return95;
}

publicinthandler(){
System.out.println("第三關(guān)-->ThirdPassHandler");
intscore=play();
if(score>=95){
//分?jǐn)?shù)>=95并且存在下一關(guān)才進(jìn)入下一關(guān)
if(this.next!=null){
returnthis.next.handler();
}
}
returnscore;
}
}

publicclassHandlerClient{
publicstaticvoidmain(String[]args){

FirstPassHandlerfirstPassHandler=newFirstPassHandler();//第一關(guān)
SecondPassHandlersecondPassHandler=newSecondPassHandler();//第二關(guān)
ThirdPassHandlerthirdPassHandler=newThirdPassHandler();//第三關(guān)

//和上面沒有更改的客戶端代碼相比,只有這里的set方法發(fā)生變化,其他都是一樣的
firstPassHandler.setNext(secondPassHandler);//第一關(guān)的下一關(guān)是第二關(guān)
secondPassHandler.setNext(thirdPassHandler);//第二關(guān)的下一關(guān)是第三關(guān)

//說明:因為第三關(guān)是最后一關(guān),因此沒有下一關(guān)

//從第一個關(guān)卡開始
firstPassHandler.handler();

}
}

責(zé)任鏈工廠改造

對于上面的請求鏈,我們也可以把這個關(guān)系維護(hù)到配置文件中或者一個枚舉中。我將使用枚舉來教會大家怎么動態(tài)的配置請求鏈并且將每個請求者形成一條調(diào)用鏈。

publicenumGatewayEnum{
//handlerId,攔截者名稱,全限定類名,preHandlerId,nextHandlerId
API_HANDLER(newGatewayEntity(1,"api接口限流","cn.dgut.design.chain_of_responsibility.GateWay.impl.ApiLimitGatewayHandler",null,2)),
BLACKLIST_HANDLER(newGatewayEntity(2,"黑名單攔截","cn.dgut.design.chain_of_responsibility.GateWay.impl.BlacklistGatewayHandler",1,3)),
SESSION_HANDLER(newGatewayEntity(3,"用戶會話攔截","cn.dgut.design.chain_of_responsibility.GateWay.impl.SessionGatewayHandler",2,null)),
;

GatewayEntitygatewayEntity;

publicGatewayEntitygetGatewayEntity(){
returngatewayEntity;
}

GatewayEnum(GatewayEntitygatewayEntity){
this.gatewayEntity=gatewayEntity;
}
}

publicclassGatewayEntity{

privateStringname;

privateStringconference;

privateIntegerhandlerId;

privateIntegerpreHandlerId;

privateIntegernextHandlerId;
}

publicinterfaceGatewayDao{

/**
*根據(jù)handlerId獲取配置項
*@paramhandlerId
*@return
*/
GatewayEntitygetGatewayEntity(IntegerhandlerId);

/**
*獲取第一個處理者
*@return
*/
GatewayEntitygetFirstGatewayEntity();
}

publicclassGatewayImplimplementsGatewayDao{

/**
*初始化,將枚舉中配置的handler初始化到map中,方便獲取
*/
privatestaticMapgatewayEntityMap=newHashMap<>();

static{
GatewayEnum[]values=GatewayEnum.values();
for(GatewayEnumvalue:values){
GatewayEntitygatewayEntity=value.getGatewayEntity();
gatewayEntityMap.put(gatewayEntity.getHandlerId(),gatewayEntity);
}
}

@Override
publicGatewayEntitygetGatewayEntity(IntegerhandlerId){
returngatewayEntityMap.get(handlerId);
}

@Override
publicGatewayEntitygetFirstGatewayEntity(){
for(Map.Entryentry:gatewayEntityMap.entrySet()){
GatewayEntityvalue=entry.getValue();
//沒有上一個handler的就是第一個
if(value.getPreHandlerId()==null){
returnvalue;
}
}
returnnull;
}
}

publicclassGatewayHandlerEnumFactory{

privatestaticGatewayDaogatewayDao=newGatewayImpl();

//提供靜態(tài)方法,獲取第一個handler
publicstaticGatewayHandlergetFirstGatewayHandler(){

GatewayEntityfirstGatewayEntity=gatewayDao.getFirstGatewayEntity();
GatewayHandlerfirstGatewayHandler=newGatewayHandler(firstGatewayEntity);
if(firstGatewayHandler==null){
returnnull;
}

GatewayEntitytempGatewayEntity=firstGatewayEntity;
IntegernextHandlerId=null;
GatewayHandlertempGatewayHandler=firstGatewayHandler;
//迭代遍歷所有handler,以及將它們鏈接起來
while((nextHandlerId=tempGatewayEntity.getNextHandlerId())!=null){
GatewayEntitygatewayEntity=gatewayDao.getGatewayEntity(nextHandlerId);
GatewayHandlergatewayHandler=newGatewayHandler(gatewayEntity);
tempGatewayHandler.setNext(gatewayHandler);
tempGatewayHandler=gatewayHandler;
tempGatewayEntity=gatewayEntity;
}
//返回第一個handler
returnfirstGatewayHandler;
}

/**
*反射實體化具體的處理者
*@paramfirstGatewayEntity
*@return
*/
privatestaticGatewayHandlernewGatewayHandler(GatewayEntityfirstGatewayEntity){
//獲取全限定類名
StringclassName=firstGatewayEntity.getConference();
try{
//根據(jù)全限定類名,加載并初始化該類,即會初始化該類的靜態(tài)段
Classclazz=Class.forName(className);
return(GatewayHandler)clazz.newInstance();
}catch(ClassNotFoundException|IllegalAccessException|InstantiationExceptione){
e.printStackTrace();
}
returnnull;
}
}

publicclassGetewayClient{
publicstaticvoidmain(String[]args){
GetewayHandlerfirstGetewayHandler=GetewayHandlerEnumFactory.getFirstGetewayHandler();
firstGetewayHandler.service();
}
}

基于微服務(wù)的思想,構(gòu)建在 B2C 電商場景下的項目實戰(zhàn)。核心技術(shù)棧,是 Spring Boot + Dubbo 。未來,會重構(gòu)成 Spring Cloud Alibaba 。

項目地址:https://github.com/YunaiV/onemall

聊聊其他

設(shè)計模式有很多,責(zé)任鏈只是其中的一種,我覺得很有意思,非常值得一學(xué)。

設(shè)計模式確實是一門藝術(shù),仍需努力呀

審核編輯 :李倩


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

    關(guān)注

    20

    文章

    2992

    瀏覽量

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

    關(guān)注

    30

    文章

    4922

    瀏覽量

    72243
  • 過濾器
    +關(guān)注

    關(guān)注

    1

    文章

    441

    瀏覽量

    20707

原文標(biāo)題:同事寫了一個責(zé)任鏈模式,bug無數(shù)!

文章出處:【微信號:AndroidPush,微信公眾號:Android編程精選】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關(guān)推薦
    熱點(diǎn)推薦

    N9H20如何將 SPI 閃存與非作系統(tǒng) BSP 一起使用?

    N9H20如何將 SPI 閃存與非作系統(tǒng) BSP 一起使用?
    發(fā)表于 09-01 08:27

    N9H20如何將非作系統(tǒng) NVTFAT 與 SPI 閃存一起使用?

    N9H20如何將非作系統(tǒng) NVTFAT 與 SPI 閃存一起使用?
    發(fā)表于 09-01 06:38

    光纖能與電線一起走嗎

    光纖與電線在特定條件下可以一起布線,但需嚴(yán)格遵守安全規(guī)范和物理隔離要求,以下是詳細(xì)分析: 、光纖與電線的物理特性差異 光纖 傳輸介質(zhì):以光信號傳輸數(shù)據(jù),不導(dǎo)電,因此不受電磁干擾(EMI
    的頭像 發(fā)表于 07-14 10:40 ?3022次閱讀

    尋開發(fā)伙伴 一起搞細(xì)胞電阻儀,有興趣的朋友來聊聊!

    尋開發(fā)伙伴 一起搞細(xì)胞電阻儀,有興趣的朋友來聊聊!
    發(fā)表于 07-10 15:51

    技術(shù)故障率降低20%,關(guān)鍵在于這套IoT軟硬件體化方案...

    設(shè)備運(yùn)行不穩(wěn)定、現(xiàn)場環(huán)境惡劣、網(wǎng)絡(luò)集成困難?一起來看看這個案例怎么破局
    的頭像 發(fā)表于 07-02 10:48 ?201次閱讀
    技術(shù)故障率降低20%,關(guān)鍵在于這套IoT軟硬件<b class='flag-5'>一</b>體化方案...

    屏蔽網(wǎng)線可以和電線一起

    屏蔽網(wǎng)線與電線不建議一起走線,原因主要有以下幾點(diǎn): 電磁干擾:電源線在傳輸電能時會產(chǎn)生電磁場,而屏蔽網(wǎng)線中的導(dǎo)線可能會受到這個電磁場的干擾。這種干擾可能導(dǎo)致屏蔽網(wǎng)線的信號質(zhì)量下降、速度變慢,甚至無法
    的頭像 發(fā)表于 03-07 10:47 ?1267次閱讀

    AN-166:與Linduino一起飛行中更新

    電子發(fā)燒友網(wǎng)站提供《AN-166:與Linduino一起飛行中更新.pdf》資料免費(fèi)下載
    發(fā)表于 01-12 10:09 ?0次下載
    AN-166:與Linduino<b class='flag-5'>一起</b>飛行中更新

    ADS5404EVM 和TSW1400EVM一起使用就可以開發(fā)嗎?

    ADS5404EVM 和TSW1400EVM一起使用就可以開發(fā)嗎? 這個開發(fā)板一起使用的時候還需要購買其他互聯(lián)電纜不? 開發(fā)套件中有沒有包含信號和時鐘的輸入同軸電纜? 三個問題。
    發(fā)表于 12-30 08:30

    和Dr Peter一起學(xué)KiCad 4.8:設(shè)計規(guī)則檢查(DRC)

    和Dr Peter一起學(xué)KiCad 4.8:設(shè)計規(guī)則檢查(DRC)
    的頭像 發(fā)表于 12-25 14:55 ?2531次閱讀
    和Dr Peter<b class='flag-5'>一起</b>學(xué)KiCad 4.8:設(shè)計規(guī)則檢查(DRC)

    快來“一起鴻蒙”!體驗更出色,智慧再升級

    當(dāng)鴻蒙遇上脫口秀,會擦出什么樣的火花?“一起鴻蒙”給出了答案。華為邀請了呼蘭、賈耗、漫才兄弟、龐博、小鹿等藝人,用脫口秀獨(dú)有的風(fēng)趣幽默方式,將原生鴻蒙帶入觀眾的視野。從真實的應(yīng)用場景出發(fā),脫口秀
    的頭像 發(fā)表于 12-25 14:45 ?563次閱讀
    快來“<b class='flag-5'>一起</b>鴻蒙<b class='flag-5'>吧</b>”!體驗更出色,智慧再升級

    將UCC39002與3個PT4484模塊一起使用

    電子發(fā)燒友網(wǎng)站提供《將UCC39002與3個PT4484模塊一起使用.pdf》資料免費(fèi)下載
    發(fā)表于 12-21 10:23 ?0次下載
    將UCC39002與3個PT4484模塊<b class='flag-5'>一起</b>使用

    如何讀取ADS1299菊花數(shù)據(jù)?

    ,菊花必須統(tǒng)使用外部時鐘。級聯(lián)模式可以分別配置不同ADS1299。 5.多片bias引腳連接如下圖(將BIASINV連接在一起,并且只留
    發(fā)表于 12-04 06:56

    ADS8327模式怎么用?

    如圖:圖59的CS是是分開的,圖60的CS時序又是同步的。CS分開很難實現(xiàn),DSP驅(qū)動時會很復(fù)雜,我想知道三片ADS8327的CS腳連一起可不可以工作。 如果CS共用,三片只能都配置為模式
    發(fā)表于 11-26 06:46

    DAC81416將多通道輸出口連一起,輸出不同電壓,會發(fā)生什么?

    大家好,如果將DAC的多個輸出通道連在一起,輸出不同電壓,會發(fā)生什么?
    發(fā)表于 11-20 07:36

    請問tas5731m PBTL模式,單聲道輸出(AB連一起,CD連一起)如何實現(xiàn)左右聲道的混音輸出?

    請問tas5731m PBTL模式,單聲道輸出(AB連一起,CD連一起)如何實現(xiàn)左右聲道的混音輸出 還有開發(fā)軟件里能直接拉線么
    發(fā)表于 10-17 06:23