Android中的设计模式之责任链模式

参考

  • 《设计模式:可复用面向对象软件的基础 》5.1 Chain of responsibility 职责链 对象行为型模式
  • 《Android源码设计模式解析与实战》第9章 使编程更有灵活性--责任链模式

意图

使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。

适用场景

  • 有多个的对象可以处理一个请求,哪个对象处理该请求运行时刻自动确定。
  • 想在不明确指定接收者的情况下,像多个对象中的一个提交一个请求。
  • 可处理一个请求的对象集合应被动态指定。

结构

结构
  • Handler 抽象处理角色,声明一个请求处理的方法,并在其中保持一个对下一个处理节点Handler对象的引用。
  • ConcreteHandler 具体处理者角色,对请求进行处理,如果不能处理则将该请求转发给下一个节点上的处理对象。

进一步抽象

对于请求来说,其形式是固定的,就是一个字符串,而判断一个节点上的对象是否能够处理该请求的标志,则是该字符串是否与之匹配。然而在大多数情况下,责任链中的请求和对应的处理规则是不尽相同的,在这种情况下可以将请求进行抽象,使之对请求的处理规则也进行封装作为一个独立的对象。

进一步抽象Request

代码实现

/**
 *  抽象处理类
 * @author newtrekWang
 * @email  wangjiaxing20160101@gmail.com
 * @time   2018/8/19  0:18
 */
public abstract class AbstractHandler {
    /**
     * 下一节点上的处理者对象
     */
    private AbstractHandler nextHandler;

    /**
     * 处理请求
     * @param request
     */
    public void handleRequest(AbstractRequest request){
        if (getHandlerLevel() == request.getRequestLevel()){
            handle(request);
        }else {
            if (nextHandler != null){
                nextHandler.handleRequest(request);
            }else {
                System.out.println("所有对象都不能处理该请求");
            }
        }
    }
    /**
     * 处理请求
     * @param request 请求对象
     */
    public abstract void handle(AbstractRequest request);

    /**
     * 每个处理者具体处理请求对象的实现
     * @return 处理级别
     */
    public abstract int getHandlerLevel();

    /**
     * 设置下一个处理者
     * @param nextHandler
     */
    public void setNextHandler(AbstractHandler nextHandler) {
        this.nextHandler = nextHandler;
    }
}

/**
 *  抽象请求类
 * @author newtrekWang
 * @email  wangjiaxing20160101@gmail.com
 * @time   2018/8/19  0:20
 */
public abstract class AbstractRequest {
    /**
     * 要处理的内容对象
     */
    private Object object;

    /**
     * 通过构造函数注入要处理的内容对象
     * @param object 要处理的内容对象
     */
    public AbstractRequest(Object object) {
        this.object = object;
    }
    /**
     * 获取要处理的内容对象
     * @return 要处理的内容对象
     */
    public Object getObject() {
        return object;
    }

    /**
     * 获取请求级别
     * @return 请求级别
     */
    public abstract int getRequestLevel();
}

**
 *  具体处理者1
 * @author newtrekWang
 * @email  wangjiaxing20160101@gmail.com
 * @time   2018/8/19  0:33
 */
public class ConcretHandler1 extends AbstractHandler {
    @Override
    public void handle(AbstractRequest request) {
        System.out.println("Handler1 handler request :"+request.getRequestLevel());
    }

    @Override
    public int getHandlerLevel() {
        return 1;
    }
}

/**
 *  具体处理者2
 * @author newtrekWang
 * @email  wangjiaxing20160101@gmail.com
 * @time   2018/8/19  0:33
 */
public class ConcretHandler2 extends AbstractHandler {
    @Override
    public void handle(AbstractRequest request) {
        System.out.println("Handler2 handler request :"+request.getRequestLevel());
    }

    @Override
    public int getHandlerLevel() {
        return 2;
    }
}

/**
 *  具体处理者3
 * @author newtrekWang
 * @email  wangjiaxing20160101@gmail.com
 * @time   2018/8/19  0:33
 */
public class ConcretHandler3 extends AbstractHandler {
    @Override
    public void handle(AbstractRequest request) {
        System.out.println("Handler3 handler request :"+request.getRequestLevel());
    }

    @Override
    public int getHandlerLevel() {
        return 3;
    }
}

/**
 *  具体请求者1
 * @author newtrekWang
 * @email  wangjiaxing20160101@gmail.com
 * @time   2018/8/19  0:36
 */
public class ConcretRequest1 extends AbstractRequest {
    /**
     * 通过构造函数注入要处理的内容对象
     *
     * @param object 要处理的内容对象
     */
    public ConcretRequest1(Object object) {
        super(object);
    }

    @Override
    public int getRequestLevel() {
        return 1;
    }
}

**
 *  具体请求者2
 * @author newtrekWang
 * @email  wangjiaxing20160101@gmail.com
 * @time   2018/8/19  0:36
 */
public class ConcretRequest2 extends AbstractRequest {
    /**
     * 通过构造函数注入要处理的内容对象
     *
     * @param object 要处理的内容对象
     */
    public ConcretRequest2(Object object) {
        super(object);
    }

    @Override
    public int getRequestLevel() {
        return 2;
    }
}

**
 *  具体请求者3
 * @author newtrekWang
 * @email  wangjiaxing20160101@gmail.com
 * @time   2018/8/19  0:36
 */
public class ConcretRequest3 extends AbstractRequest {
    /**
     * 通过构造函数注入要处理的内容对象
     *
     * @param object 要处理的内容对象
     */
    public ConcretRequest3(Object object) {
        super(object);
    }

    @Override
    public int getRequestLevel() {
        return 3;
    }
}


 public static void main(String[] args){
        // 三个不同级别的请求
        AbstractRequest request1 = new ConcretRequest1("request 1");
        AbstractRequest  request2 = new ConcretRequest2("request 2");
        AbstractRequest  request3 = new ConcretRequest3("request 3");
        //三个不同级别的处理者
        AbstractHandler handler1 = new ConcretHandler1();
        AbstractHandler handler2 = new ConcretHandler2();
        AbstractHandler handler3 = new ConcretHandler3();
        // 设置处理者的链式关系
        handler1.setNextHandler(handler2);
        handler2.setNextHandler(handler3);
        // 总是从链子的首端发起请求
        handler1.handleRequest(request1);
        handler1.handleRequest(request2);
        handler1.handleRequest(request3);
    }
    

输出结果:

Handler1 handler request :1
Handler2 handler request :2
Handler3 handler request :3

优点

对请求者和处理者关系解耦,提高代码的灵活性。

缺点

如果处理者太多,那么遍历必定会影响性能。

应用例子1 Android 的触摸事件传递与分发机制

应用例子2 利用有序广播实现责任链事件处理

Android种的BroastCast分为两种,一种时普通广播,另一种是有序广播。普通广播是异步的,发出时可以被所有的接收者收到。而有序广播是根据优先级一次传播的,直到有接收者将其终止或者所有接收者都不终止它。有序广播的这一特性与我们的责任链模式很相近,我们可以轻松地实现一种全局的责任链事件处理。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,026评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,655评论 1 296
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,726评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,204评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,558评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,731评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,944评论 2 314
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,698评论 0 203
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,438评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,633评论 2 247
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,125评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,444评论 3 255
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,137评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,103评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,888评论 0 197
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,772评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,669评论 2 271

推荐阅读更多精彩内容

  • 责任链模式的介绍 ​ 责任链模式(Iterator Pattren),是行为型设计模式之一。什么是“链”?我们...
    Android天之骄子阅读 694评论 0 1
  • 妹妹去赏牡丹,发来很多照片,知我业余爱好,给我写画素材。 灯下观叶赏花,碟中调制墨色,铺纸画出扇面,试绘富贵花儿。
    品茶轩pcx阅读 199评论 4 7
  • 生活方式 换洗了床单被罩 大扫除了卫生 热水洗澡 回归生活本质 经过多角色锤炼 才懂得缩减角色很重要 专攻所长才能...
    舍得_之间阅读 153评论 1 5