責(zé)任鏈模式(Chain of Responsibility Pattern)

責(zé)任鏈模式是一種對象的行為模式。在責(zé)任鏈模式里,很多對象由每一個對象對其下家的引用而連接起來形成一條鏈。請求在這個鏈上傳遞,直到鏈上的某一個對象決定處理此請求。發(fā)出這個請求的客戶端并不知道鏈上的哪一個對象最終處理這個請求,這使得系統(tǒng)可以在不影響客戶端的情況下動態(tài)地重新組織和分配責(zé)任。 -《JAVA與模式》

1. 什么是責(zé)任鏈

在《HeadFisrt設(shè)計模式》一書中并沒有關(guān)于責(zé)任鏈描述,再次驗證設(shè)計模式只是編碼的約定,某個通俗易懂的名稱,并不是某種蓋棺定論的東西。責(zé)任鏈模式和觀察者模式有相似的的地方,即事件(或責(zé)任)的傳遞。觀察者是發(fā)散式的,一對多;責(zé)任鏈?zhǔn)擎準(zhǔn)降模粋饕弧?/p>

/* 抽象的責(zé)任處理者 */

public abstract class AbstractHandler {

    private AbstractHandler next;

    public void handle(String param) {
        /* 做自己的事情 讓別人無事可做 */
        customized(param);

        if (next != null) {
            next.handle(param);
        }
    }

    public abstract void customized(String param);

    public void setNext(AbstractHandler next) {
        this.next = next;
    }
}
/* 具體的責(zé)任處理者 */

public class Handler1 extends AbstractHandler {

    @Override
    public void customized(String param) {
        System.out.println("handler1: handle " + param);
    }
}

public class Handler2 extends AbstractHandler {

    @Override
    public void customized(String param) {
        System.out.println("handler2: handle " + param);
    }
}

public class Handler3 extends AbstractHandler {

    @Override
    public void customized(String param) {
        System.out.println("handler3: handle " + param);
    }
}
/* 當(dāng)有大事要發(fā)生的時候 */

public class HandlerMain {

    public static void main(String[] args) {
        AbstractHandler chain = HandlerChainFactory.generate();
        chain.handle("rabbit");
    }
}

public class HandlerChainFactory {

    public static AbstractHandler generate() {
        AbstractHandler chain = new Handler1();
        Handler2 handler2 = new Handler2();
        Handler3 handler3 = new Handler3();

        chain.setNext(handler2);
        handler2.setNext(handler3);

        return chain;
    }
}

2. 為什么要用責(zé)任鏈

解耦:模塊間解耦,各模塊負(fù)責(zé)各自的業(yè)務(wù),職責(zé)明確單一
有序:模塊有序排布、執(zhí)行,方便跳轉(zhuǎn)
可擴(kuò):易于擴(kuò)展,項鏈表一樣快速插入

3. 怎么使用責(zé)任鏈

責(zé)任鏈在使用方式上,按控制邏輯的位置可以分為:內(nèi)控和外控。

3.1 內(nèi)控

內(nèi)控是指控制邏輯在模塊內(nèi)部,由各模塊來控制責(zé)任鏈的走向:能不能繼續(xù)走下去。代表有:Tomcat

Tomcat 中對 Servlet Filter 的處理
Tomcat
Tomcat
3.2 外控

外控是指控制邏輯在模塊外部,各模塊只負(fù)責(zé)各自的業(yè)務(wù),模塊跳轉(zhuǎn)邏輯單獨(dú)處理。代表有:Spring MVC Interceptor 和 Zuul

  1. Spring MVC

![(https://upload-images.jianshu.io/upload_images/3596970-162b94119bbeae23.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

洋蔥圈模型
DispatcherServlet
HandlerExecutionChain
  1. Zuul
/*
 * Copyright 2013 Netflix, Inc.
 *
 *      Licensed under the Apache License, Version 2.0 (the "License");
 *      you may not use this file except in compliance with the License.
 *      You may obtain a copy of the License at
 *
 *          http://www.apache.org/licenses/LICENSE-2.0
 *
 *      Unless required by applicable law or agreed to in writing, software
 *      distributed under the License is distributed on an "AS IS" BASIS,
 *      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 *      See the License for the specific language governing permissions and
 *      limitations under the License.
 */
package com.netflix.zuul.http;

import com.netflix.zuul.FilterProcessor;
import com.netflix.zuul.ZuulRunner;
import com.netflix.zuul.context.RequestContext;
import com.netflix.zuul.exception.ZuulException;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;
import org.mockito.runners.MockitoJUnitRunner;

import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;

import static org.junit.Assert.assertTrue;
import static org.mockito.Mockito.*;

/**
 * Core Zuul servlet which intializes and orchestrates zuulFilter execution
 *
 * @author Mikey Cohen
 *         Date: 12/23/11
 *         Time: 10:44 AM
 */
public class ZuulServlet extends HttpServlet {

    private static final long serialVersionUID = -3374242278843351500L;
    private ZuulRunner zuulRunner;


    @Override
    public void init(ServletConfig config) throws ServletException {
        super.init(config);

        String bufferReqsStr = config.getInitParameter("buffer-requests");
        boolean bufferReqs = bufferReqsStr != null && bufferReqsStr.equals("true") ? true : false;

        zuulRunner = new ZuulRunner(bufferReqs);
    }

    @Override
    public void service(javax.servlet.ServletRequest servletRequest, javax.servlet.ServletResponse servletResponse) throws ServletException, IOException {
        try {
            init((HttpServletRequest) servletRequest, (HttpServletResponse) servletResponse);

            // Marks this request as having passed through the "Zuul engine", as opposed to servlets
            // explicitly bound in web.xml, for which requests will not have the same data attached
            RequestContext context = RequestContext.getCurrentContext();
            context.setZuulEngineRan();

            try {
                preRoute();
            } catch (ZuulException e) {
                error(e);
                postRoute();
                return;
            }
            try {
                route();
            } catch (ZuulException e) {
                error(e);
                postRoute();
                return;
            }
            try {
                postRoute();
            } catch (ZuulException e) {
                error(e);
                return;
            }

        } catch (Throwable e) {
            error(new ZuulException(e, 500, "UNHANDLED_EXCEPTION_" + e.getClass().getName()));
        } finally {
            RequestContext.getCurrentContext().unset();
        }
    }

    /**
     * executes "post" ZuulFilters
     *
     * @throws ZuulException
     */
    void postRoute() throws ZuulException {
        zuulRunner.postRoute();
    }

    /**
     * executes "route" filters
     *
     * @throws ZuulException
     */
    void route() throws ZuulException {
        zuulRunner.route();
    }

    /**
     * executes "pre" filters
     *
     * @throws ZuulException
     */
    void preRoute() throws ZuulException {
        zuulRunner.preRoute();
    }

    /**
     * initializes request
     *
     * @param servletRequest
     * @param servletResponse
     */
    void init(HttpServletRequest servletRequest, HttpServletResponse servletResponse) {
        zuulRunner.init(servletRequest, servletResponse);
    }

    /**
     * sets error context info and executes "error" filters
     *
     * @param e
     */
    void error(ZuulException e) {
        RequestContext.getCurrentContext().setThrowable(e);
        zuulRunner.error();
    }

}

4. 怎么選用責(zé)任鏈實(shí)現(xiàn)

洋蔥式

代表:Spring MVC Interceptor
優(yōu)點(diǎn):Interceptor 在方法界別分隔預(yù)處理、后置處理邏輯,職責(zé)明確
缺點(diǎn):大部分 Interceptor 只包含預(yù)處理或只包含后置處理,Interceptor 相對較重

半開式

代表:Zuul Filter
優(yōu)點(diǎn):輕量靈活,符合大部分業(yè)務(wù)處理邏輯(相對洋蔥式)
缺點(diǎn):不方便實(shí)現(xiàn)環(huán)裝邏輯(相對洋蔥式)

鏈表式

代表:Pigeon Filter 和 Servlet Filter
優(yōu)點(diǎn):支持環(huán)裝結(jié)構(gòu),可以實(shí)現(xiàn) try-catch 邏輯,控制更加靈活
缺點(diǎn):如果需要開放給外部使用,出錯風(fēng)險相對較高

一句話總結(jié):

  • 開放給外部業(yè)務(wù)使用,推薦洋蔥式
  • 內(nèi)部業(yè)務(wù)使用,推薦鏈表式
  • 復(fù)雜的 Servlet 的邏輯,推薦半開式
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,316評論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,481評論 3 415
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,241評論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,939評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,697評論 6 409
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,182評論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,247評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,406評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,933評論 1 334
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,772評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,973評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,516評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,209評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,638評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,866評論 1 285
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,644評論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,953評論 2 373

推薦閱讀更多精彩內(nèi)容