文档章节

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

newtrek
 newtrek
发布于 2018/08/19 10:52
字数 1355
阅读 9
收藏 1

参考

  • 《设计模式:可复用面向对象软件的基础 》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分为两种,一种时普通广播,另一种是有序广播。普通广播是异步的,发出时可以被所有的接收者收到。而有序广播是根据优先级一次传播的,直到有接收者将其终止或者所有接收者都不终止它。有序广播的这一特性与我们的责任链模式很相近,我们可以轻松地实现一种全局的责任链事件处理。

© 著作权归作者所有

newtrek
粉丝 3
博文 21
码字总数 35117
作品 0
江北
程序员
私信 提问
android 设计模式的应用

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

小克898
2014/08/24
0
0
Android 网络编程 目录

Android 网络编程 目录 Android 网络编程1 Http协议 Android 网络编程2 Okhttp缓存机制 Android 网络编程3 Java NIO to be continued... Android 架构师之路 目录 Android 架构师之路1 UML图...

香沙小熊
2018/06/21
0
0
Android--面试中遇到的问题总结(三)

《Android 开发工程师面试指南 LearningNotes 》,作者是陶程,由梁观全贡献部分。大家可以去知乎关注这两位用心的少年。这份指南包含了大部分Android开发的基础、进阶知识,不仅可以帮助准备...

sealin
2017/02/22
0
0
Tomcat 系统架构与设计模式_ 设计模式分析

门面设计模式 门面设计模式在 Tomcat 中有多处使用,在 Request 和 Response 对象封装中、Standard Wrapper 到 ServletConfig 封装中、ApplicationContext 到 ServletContext 封装中等都用到...

lvzjane
2014/11/03
0
0
设计模式知识汇总(附github分享)

写在前面 主要内容 为了更系统的学习设计模式,特地开了这样一个基于Java的设计模式【集中营】,都是笔者在实际工作中用到过或者学习过的一些设计模式的一些提炼或者总检。慢慢地初见规模,也...

landy8530
2018/10/10
0
0

没有更多内容

加载失败,请刷新页面

加载更多

八、Docker Swarm

Docker Swarm有两件事:一个企业级的Docker主机安全集群,另一个是用于协调微服务应用程序的引擎。 在集群方面,它将一个或多个Docker节点组合在一起,并允许你将他们作为一个集群来管理。开...

倪伟伟
昨天
4
0
Fragment懒加载其实很简单

前言 记得去年面试的时候, 面了一家小公司, 那个面试官问我, fragment的懒加载做过吗?我说没做过(确实没做过).后来面试快结束了, 又问我, 懒加载没做过是吗?后来可想而知也没收到offer, (ಥ_...

天王盖地虎626
昨天
2
0
聊聊dubbo的TimeoutFilter

序 本文主要研究一下dubbo的TimeoutFilter ListenableFilter dubbo-2.7.2/dubbo-rpc/dubbo-rpc-api/src/main/java/org/apache/dubbo/rpc/ListenableFilter.java public abstract class Liste......

go4it
昨天
6
0
方法与数组

方法 方法就是完成特定功能的代码块;在很多语言里面都有函数的定义,函数在Java中被称为方法 格式: 修饰符 返回值类型 方法名(参数类型 参数名1,参数类型 参数名2…) throw 异常{ 函数体;...

凹凸凸
昨天
4
0
死磕 java同步系列之StampedLock源码解析

问题 (1)StampedLock是什么? (2)StampedLock具有什么特性? (3)StampedLock是否支持可重入? (4)StampedLock与ReentrantReadWriteLock的对比? 简介 StampedLock是java8中新增的类,...

彤哥读源码
昨天
16
1

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部