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

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

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

什么是責(zé)任鏈?

汽車電子技術(shù) ? 來源:程序猿知秋 ? 作者: 程序猿知秋 ? 2023-02-16 14:41 ? 次閱讀

什么是責(zé)任鏈?

責(zé)任鏈模式是行為模式的一種,它將需要觸發(fā)的Handler組成一條鏈,發(fā)送者將請(qǐng)求發(fā)給鏈的第一個(gè)接收者,并且沿著這條鏈傳遞,直到有一個(gè)Handler來處理它或者直到最后也沒有對(duì)象處理而留在鏈末尾端;責(zé)任連模式的重點(diǎn)是在鏈上,由一條鏈去處理相似的請(qǐng)求,在鏈中決定誰來處理這個(gè)請(qǐng)求

責(zé)任鏈分為純責(zé)任鏈與不純責(zé)任鏈(一般實(shí)戰(zhàn)應(yīng)用較多)

**純職責(zé)鏈 **

1.每個(gè)處理者接收到請(qǐng)求后,要么單純轉(zhuǎn)發(fā)請(qǐng)求,要么單純處理請(qǐng)求。不允許既處理請(qǐng)求,又轉(zhuǎn)發(fā)請(qǐng)求的情況。

2.請(qǐng)求必須被責(zé)任鏈上的某個(gè)處理者處理。不允許出現(xiàn)請(qǐng)求未被處理的情況。

**不純職責(zé)鏈 **

1.每個(gè)處理者接收到請(qǐng)求后,除了單純轉(zhuǎn)發(fā)請(qǐng)求,或者單純處理請(qǐng)求,還可以部分處理請(qǐng)求后,轉(zhuǎn)發(fā)。

2.請(qǐng)求可以不被責(zé)任鏈上的任何處理者處理。

責(zé)任鏈優(yōu)點(diǎn)

1.解耦請(qǐng)求者和處理者。

2.非常靈活,可以任意組裝

3.各個(gè)節(jié)點(diǎn)的責(zé)任明確

缺點(diǎn)

1.每次都是從鏈頭開始。

2. 可能造成死循環(huán)。責(zé)任鏈如果是環(huán)狀的,可能會(huì)導(dǎo)致循環(huán)調(diào)用,造成死循環(huán)。

純責(zé)任鏈的代碼Demo:

/**
     * 在公司OA系統(tǒng)請(qǐng)假審批流程
如果請(qǐng)假小于3天只需要項(xiàng)目經(jīng)理批復(fù)就行;
如果請(qǐng)假大于等于3天,小于7天需要人事經(jīng)理批復(fù)了;
如果請(qǐng)假大于等于7天,小于15天需要總經(jīng)理批復(fù)了;
如果申請(qǐng)請(qǐng)假大于等于15天,決絕批復(fù)......
     * @param args
     */
 public static void main(String[] args) {
        Leader manager = new Manager("張三");
        Leader hrManager = new HrManager("李四");
        Leader generalManager = new GeneralManager("王麻子");


        //組織責(zé)任鏈對(duì)象的關(guān)系
        manager.setNextLeader(hrManager);
        hrManager.setNextLeader(generalManager);
        //請(qǐng)假
        LeaveRequest request = new LeaveRequest("parry", 10, "回家相親!");
        manager.dealLeaveRequest(request);
  }

public abstract class Leader {
    // 領(lǐng)導(dǎo)姓名
    protected String name;
    // 責(zé)任鏈上的后繼對(duì)象
    protected Leader nextLeader;
    public Leader(String name) {
        super();
        this.name = name;
    }


    public void setNextLeader(Leader nextLeader) {
        this.nextLeader = nextLeader;
    }

    public abstract void dealLeaveRequest(LeaveRequest request);
}

  public class HrManager extends Leader{
      public HrManager(String name) {
          super(name);
      }


      @Override
      public void dealLeaveRequest(LeaveRequest request) {
        if (3 <= request.getLeaveDay() && request.getLeaveDay() < 7) {
            System.out.println("人事經(jīng)理:" + name + " 審批了 " + request.getEmployee() + "請(qǐng)假" + request.getLeaveDay()
                    + "天的請(qǐng)求,請(qǐng)假原因:" + request.getReason());
        } else {
            if (this.nextLeader != null) {
                this.nextLeader.dealLeaveRequest(request);
            }
        }
    }
}


public class GeneralManager extends Leader {
    public GeneralManager(String name) {
        super(name);
    }


    @Override
    public void dealLeaveRequest(LeaveRequest request) {
        if (7 <= request.getLeaveDay() && request.getLeaveDay() <= 15) {
            System.out.println("總經(jīng)理:" + name + " 審批了 " + request.getEmployee() + "請(qǐng)假" + request.getLeaveDay()
                    + "天的請(qǐng)求,請(qǐng)假原因:" + request.getReason());
        } else {
            System.out.println(
                    "總經(jīng)理:" + name + " 拒絕了 " + request.getEmployee() + "請(qǐng)假" + request.getLeaveDay() + "天的請(qǐng)求,請(qǐng)假不能超過15天");
        }
    }
}


public class Manager extends Leader{
    public Manager(String name) {
        super(name);
    }


    @Override
    public void dealLeaveRequest(LeaveRequest request) {
        if (request.getLeaveDay() < 3) {
            System.out.println("經(jīng)理:" + name + " 審批了 " + request.getEmployee() + "請(qǐng)假" + request.getLeaveDay()
                    + "天的請(qǐng)求,請(qǐng)假原因:" + request.getReason());
        } else {
            if (this.nextLeader != null) {
                this.nextLeader.dealLeaveRequest(request);
            }
        }
    }
}


@Data
public class LeaveRequest {
   //姓名
    private String name;
   //請(qǐng)假天數(shù)
    private int leaveDay;
    //請(qǐng)假原因
    private String reason;


    public LeaveRequest(String name, int leaveDay, String reason) {
        this.name=name;
        this.leaveDay=leaveDay;
        this.reason=reason;
    }  
}

電商售后使用責(zé)任鏈(不純責(zé)任鏈)

售后業(yè)務(wù)梳理

電商售后常見三種模式:

僅退款、退款退貨、退款換貨

售后業(yè)務(wù)可分為

僅退款: 采購(gòu)商申請(qǐng)——>商家審核——>完成

退款退貨:采購(gòu)商申請(qǐng)——>商家審核——>采購(gòu)商發(fā)貨——>商家確認(rèn)收貨——>完成

退款換貨:采購(gòu)商申請(qǐng)——>商家審核——>采購(gòu)商發(fā)貨——>商家確認(rèn)收貨——>商家發(fā)出新品——>采購(gòu)商確認(rèn)收貨——>完成

僅退款的實(shí)際退款等操作在商家審核時(shí),而退款退貨和退款換貨則在確認(rèn)收貨時(shí)。

以下為售后責(zé)任鏈流程:

售后->校驗(yàn)退款單狀態(tài)->原路退款(以下單支付方式為準(zhǔn))->退積分或其它優(yōu)惠->更新訂單及對(duì)應(yīng)商品退款信息->保存結(jié)算->保存財(cái)務(wù)流水->保存/更新退款日志->發(fā)送退款成功異步消息

三種售后的核心退款業(yè)務(wù)流程基本是一致的,少個(gè)別節(jié)點(diǎn)業(yè)務(wù)邏輯有所區(qū)別,將核心業(yè)務(wù)抽成責(zé)任鏈的各個(gè)節(jié)點(diǎn),這對(duì)售后來說,代碼的復(fù)用性提高了很多,同時(shí)業(yè)務(wù)處理更加清晰,犯錯(cuò)率大大降低。

以下為售后中代碼使用Demo

定義一個(gè)節(jié)點(diǎn)的基類接口

public interface BaseRefundHandler {
    void handle(PipelineContext context); 
}

售后責(zé)任鏈組裝,以下以僅退款為例

public class RefundPipeBean {
/**
 * 僅退款,責(zé)任鏈初始化節(jié)點(diǎn)
 */
public List<String> refundPipe = new ArrayList<>();


  public List<String> getRefundPipe() {
      refundPipe.add("checkRefundStatus"); //校驗(yàn)退款單狀態(tài)
      refundPipe.add("refundMoney");  //原路退款(獲取訂單支付方式)
      refundPipe.add("returnScore");  //退積分或其它優(yōu)惠券類
       ......
      return refundPipe;
  }


  @Bean
  @Scope("prototype")
  public CheckRefundStatus checkRefundStatus() {return new CheckRefundStatus();}

  @Bean
  @Scope("prototype")
  public RefundMoney refundMoney() {return new RefundMoney();}
  ......
}

** 注: 這個(gè)配置就等同于之前在xml里的配置**

"checkRefundStatus" class="com.xx.CheckRefundStatus"/>

創(chuàng)建一個(gè)啟動(dòng)監(jiān)聽器

@Component
public class RefundPipeListener extends ApplicationObjectSupport implements InitializingBean {
    private static Map<String, List> handlersMap = new HashMap<>();
     
   public static Map<String, List> getHandlersMap() {
        return handlersMap;
    }

    //最好定義成全局的,此處因演示 才在類中寫 。 責(zé)任鏈的名稱
   public static String ONLY_REFUND="ONLY_REFUND";
  @Override
  public void afterPropertiesSet() throws Exception {
    RefundPipeBean refundPipeBean=new RefundPipeBean();
     //裝載退款業(yè)務(wù)鏈.....
    List normalHandlers=new ArrayList<>();

    for (String s : refundPipeBean.getRefundPipe()) {
        BaseRefundHandler normalHandler = (BaseRefundHandler) getApplicationContext().getBean(s);
        normalHandlers.add(normalHandler);
        handlersMap.put(ONLY_REFUND, normalHandlers);
    } 
  }
}

創(chuàng)建節(jié)點(diǎn)傳遞對(duì)象所用POJO類

@Data
public class PipelineContext<K, V> extends ConcurrentHashMap<K, V> {
    protected boolean success = false;
    //責(zé)任鏈結(jié)束標(biāo)識(shí)
    protected boolean isEnd = false;
    protected String resultCode;
    private  Object data;


    public boolean isSuccess() {
        return success;
    }
    public void setSuccess(boolean success) {
        this.success = success;
    }


    public boolean isEnd() {
        return isEnd;
    }


    public void setEnd(boolean end) {
        isEnd = end;
    }
}


public class RefundPipelineContext<K, V> extends PipelineContext<K, V> {
    /**
     * 封裝退款單信息
    */
    private RefundInfoPipeVo refundInfoPipeVo;


    public RefundInfoPipeVo getRefundInfoPipeVo() {
        return refundInfoPipeVo;
    }


    public void setRefundInfoPipeVo(RefundInfoPipeVo refundInfoPipeVo) {
        this.refundInfoPipeVo = refundInfoPipeVo;
    }
}

創(chuàng)建節(jié)點(diǎn)

public class CheckRefundStatus implements BaseRefundHandler {
  @Override
  public void handle(PipelineContext<String, RefundInfoPipeVo> context) {
     //業(yè)務(wù)邏輯 todo
     ......
   //表示該節(jié)點(diǎn)運(yùn)行正常,可以繼續(xù)向下走
   context.setSuccess(true);
  }
}

業(yè)務(wù)調(diào)用層

@RestController
public class RefundInfoController {
  @Autowired
  RefundInfoService refundInfoService;

  @ApiOperation("僅退款")
  @RequestMapping(value = "/auditRefund", method = {RequestMethod.GET, RequestMethod.POST}, produces = "application/json;charset=UTF-8")
  public RespData<Boolean> auditRefund(xxx) {
      refundInfoService.auditRefund(xxx);
      return RespData.success("審核成功",true);
  }
}


public interface RefundInfoService {
  void auditRefund(xxx);
}


@Service("refundInfoService")
public class RefundInfoServiceImpl implements RefundInfoService {


    @Override
    public void auditRefund(xxx) {
        //業(yè)務(wù)邏輯 todo 
        RefundInfoResultVo refundInfoResultVo = refundInfoMapper.queryById(platformId, refundId);
        ......
        PipelineContext<String, RefundInfoPipeVo> refund = new RefundPipelineContext<>();
        //在責(zé)任鏈中傳遞參數(shù)對(duì)象
        refund.put("refund", refundInfoResultVo);
        Map<String, List> map = RefundPipelineChangeListener.getHandlersMap();
        //通過責(zé)任鏈的名稱獲取對(duì)應(yīng)鏈 。 此處key值應(yīng)為一個(gè)全局的常量,與上面監(jiān)聽器中的鏈條名一致
        List baseRefundHandlers = map .get(ONLY_REFUND); 
        for (BaseRefundHandler handler : baseRefundHandlers) {
            handler.handle(refund);
            if (refund.isEnd()) {
                System.out.println("責(zé)任鏈執(zhí)行結(jié)束...");
                break;
            }
            if (!refund.isSuccess()) {
                        System.out.println("責(zé)任鏈執(zhí)行失敗...");
                break;
            }
        }
    }
}

整體總結(jié)

以上是責(zé)任鏈在業(yè)務(wù)中創(chuàng)建及使用的流程。

同時(shí)以上流程也還存在一些可優(yōu)化點(diǎn):

  1. 責(zé)任鏈節(jié)點(diǎn)硬編碼問題
  2. 添加同步事務(wù)控制
聲明:本文內(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)投訴
  • 模式
    +關(guān)注

    關(guān)注

    0

    文章

    65

    瀏覽量

    13401
  • handler
    +關(guān)注

    關(guān)注

    0

    文章

    7

    瀏覽量

    3035
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    Spring高級(jí)49講-060-模擬實(shí)現(xiàn)調(diào)用-責(zé)任模式

    spring
    電子學(xué)習(xí)
    發(fā)布于 :2023年01月07日 17:55:22

    基于ArkUI框架開發(fā)-ImageKnife渲染層重構(gòu)

    的實(shí)現(xiàn),都采用了責(zé)任模式,提高了自定義繪制擴(kuò)展能力。將部分通用方法封裝成工廠方法,減少開發(fā)者代碼量。通用方法從配置參數(shù)剝離,可采用鏈?zhǔn)秸{(diào)用方式使用這些方法。為了支持列表ImageKnifeOption
    發(fā)表于 04-06 10:01

    電子行業(yè)供應(yīng)責(zé)任向前邁出一步

    電子行業(yè)供應(yīng)責(zé)任向前邁出一步     瑞士蒙特勒2010年3月11日電 /美通社亞洲/ -- Achilles Information Ltd. 與 Global e-Sustainability Initiati
    發(fā)表于 03-11 18:19 ?468次閱讀

    5G時(shí)代的關(guān)鍵安全特征介紹

    同時(shí),云計(jì)算也在推動(dòng)業(yè)務(wù)的發(fā)展,讓供應(yīng)責(zé)任日益的復(fù)雜。使用電信的運(yùn)營(yíng)商各式各樣的豐富的服務(wù)的同時(shí),也帶來了挑戰(zhàn)。其解決之道是通過API來提供,保證設(shè)計(jì)的安全。Jon France指出,“這不
    發(fā)表于 05-09 09:14 ?1709次閱讀

    精通Python設(shè)計(jì)模式的PDF電子書免費(fèi)下載

    本書分三部分、共 16 章,介紹一些常用的設(shè)計(jì)模式。第一部分介紹處理對(duì)象創(chuàng)建的設(shè)計(jì)模式,包括工廠模式、建造者模式、原型模式;第二部分介紹處理一個(gè)系統(tǒng)中不同實(shí)體(類、對(duì)象等)之間關(guān)系的設(shè)計(jì)模式,包括外觀模式、享元模式等 ;第三部分介紹處理系統(tǒng)實(shí)體之間通信的設(shè)計(jì)模式,包括責(zé)任
    發(fā)表于 06-13 08:00 ?9次下載

    FCA加入RSBN合作項(xiàng)目,為電動(dòng)汽車建立負(fù)責(zé)任的鋰離子電池供應(yīng)

    據(jù)外媒報(bào)道,當(dāng)?shù)貢r(shí)間12月10日,F(xiàn)CA宣布加入負(fù)責(zé)任采購(gòu)區(qū)塊網(wǎng)絡(luò)(Responsible Sourcing Blockchain Network,RSBN),后者是一個(gè)合作項(xiàng)目,主要利用區(qū)塊技術(shù),實(shí)現(xiàn)從礦場(chǎng)到市場(chǎng)的可持續(xù)、
    的頭像 發(fā)表于 12-12 17:01 ?3871次閱讀

    C語言設(shè)計(jì)模式的程序資料合集

    之模板模式,C語言設(shè)計(jì)模式之工廠模式,C語言設(shè)計(jì)模式之責(zé)任模式,C語言設(shè)計(jì)模式之抽象工廠模式,C語言設(shè)計(jì)模式之狀態(tài)模式。
    發(fā)表于 11-16 08:00 ?5次下載

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

    如何解決這個(gè)問題,我們可以通過鏈表將每一關(guān)連接起來,形成責(zé)任的方式,第一關(guān)通過后是第二關(guān),第二關(guān)通過后是第三關(guān) ....,這樣客戶端就不需要進(jìn)行多重 if 的判斷了
    的頭像 發(fā)表于 07-08 16:25 ?874次閱讀

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

    那么有什么更好的參數(shù)校驗(yàn)的方式呢?本文就推薦一種通過責(zé)任設(shè)計(jì)模式來優(yōu)雅地實(shí)現(xiàn)參數(shù)的校驗(yàn)功能,我們通過一個(gè)用戶注冊(cè)的例子來講明白如何實(shí)現(xiàn)。
    的頭像 發(fā)表于 04-06 15:00 ?465次閱讀

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

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

    設(shè)計(jì)模式行為型:責(zé)任模式

    將請(qǐng)求發(fā)送者和請(qǐng)求接受者解耦,讓請(qǐng)求的接受者形成鏈?zhǔn)讲僮鳎腥硕寄軌蚪邮芙邮艿秸?qǐng)求,直到有人處理請(qǐng)求。
    的頭像 發(fā)表于 06-06 17:33 ?781次閱讀
    設(shè)計(jì)模式行為型:<b class='flag-5'>責(zé)任</b><b class='flag-5'>鏈</b>模式

    AMD借助創(chuàng)新與合作推進(jìn)企業(yè)責(zé)任

    、數(shù)字化影響、供應(yīng)責(zé)任,以及多元化、歸屬感和包容性等方面取得的進(jìn)展。AMD已連續(xù)28年發(fā)布其企業(yè)責(zé)任項(xiàng)目及計(jì)劃,今年的報(bào)告首次包含近期收購(gòu)公司的環(huán)境和社會(huì)數(shù)據(jù)。 AMD企業(yè)責(zé)任與國(guó)際
    的頭像 發(fā)表于 08-25 12:58 ?352次閱讀

    AMD發(fā)布年度企業(yè)責(zé)任報(bào)告 助力解決全球最重要的挑戰(zhàn)

    ? ? AMD的節(jié)能、高性能產(chǎn)品和深度合作 助力解決全球最重要的挑戰(zhàn) AMD發(fā)布年度企業(yè)責(zé)任報(bào)告,詳細(xì)闡述了公司在環(huán)境可持續(xù)發(fā)展、數(shù)字化影響、供應(yīng)責(zé)任,以及多元化、歸屬感和包容性等方面取得的進(jìn)展
    的頭像 發(fā)表于 09-07 16:20 ?1109次閱讀

    設(shè)計(jì)模式之責(zé)任模式概述

    設(shè)計(jì)模式是一些被反復(fù)使用的、具有普遍性的設(shè)計(jì)解決方案,它們是在特定情境下對(duì)軟件設(shè)計(jì)問題的成功解決方式的總結(jié)和歸納。
    的頭像 發(fā)表于 09-27 09:54 ?696次閱讀
    設(shè)計(jì)模式之<b class='flag-5'>責(zé)任</b><b class='flag-5'>鏈</b>模式概述

    還在自己實(shí)現(xiàn)責(zé)任?我建議你造輪子之前先看看這個(gè)開源項(xiàng)目

    1. 前言 設(shè)計(jì)模式在軟件開發(fā)中被廣泛使用。通過使用設(shè)計(jì)模式,開發(fā)人員可以更加高效地開發(fā)出高質(zhì)量的軟件系統(tǒng),提高代碼的可讀性、可維護(hù)性和可擴(kuò)展性。 責(zé)任模式是一種常用的行為型設(shè)計(jì)模式,它將請(qǐng)求沿著
    的頭像 發(fā)表于 09-20 14:38 ?375次閱讀
    還在自己實(shí)現(xiàn)<b class='flag-5'>責(zé)任</b><b class='flag-5'>鏈</b>?我建議你造輪子之前先看看這個(gè)開源項(xiàng)目