深入理解JavaScript系列(38):设计模式之职责链模式详解


Posted in Javascript onMarch 04, 2015

介绍

职责链模式(Chain of responsibility)是使多个对象都有机会处理请求,从而避免请求的发送者和接受者之间的耦合关系。将这个对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理他为止。

也就是说,请求以后,从第一个对象开始,链中收到请求的对象要么亲自处理它,要么转发给链中的下一个候选者。提交请求的对象并不明确知道哪一个对象将会处理它——也就是该请求有一个隐式的接受者(implicit receiver)。根据运行时刻,任一候选者都可以响应相应的请求,候选者的数目是任意的,你可以在运行时刻决定哪些候选者参与到链中。

正文

对于JavaScript实现,我们可以利用其原型特性来实现职责链模式。

var NO_TOPIC = -1;

var Topic;
function Handler(s, t) {

    this.successor = s || null;

    this.topic = t || 0;

}
Handler.prototype = {

    handle: function () {

        if (this.successor) {

            this.successor.handle()

        }

    },

    has: function () {

        return this.topic != NO_TOPIC;

    }

};

Handler只是接受2个参数,第一个是继任者(用于将处理请求传下去),第二个是传递层级(可以用于控制在某个层级下是否执行某个操作,也可以不用),Handler原型暴露了一个handle方法,这是实现该模式的重点,先来看看如何使用上述代码。
var app = new Handler({

        handle: function () {

            console.log('app handle');

        }

    }, 3);
    var dialog = new Handler(app, 1);
    var button = new Handler(dialog, 2);
    button.handle();

改代码通过原型特性,调用代码从button.handle()->dialog.handle()->app.handle()->参数里的handle(),前三个都是调用原型的handle,最后才查找到传入的参数里的handle,然后输出结果,也就是说其实只有最后一层才处理。

那如何做到调用的时候,只让dialog的这个对象进行处理呢?其实可以定义dialog实例对象的handle方法就可以了,但需要在new button的之前来做,代码如下:

var app = new Handler({

        handle: function () {

            console.log('app handle');

        }

    }, 3);
    var dialog = new Handler(app, 1);

    dialog.handle = function () {

        console.log('dialog before ...')

        // 这里做具体的处理操作

        console.log('dialog after ...')

    };
    var button = new Handler(dialog, 2);
    button.handle();

该代码的执行结果即时dialog.handle里的处理结果,而不再是给app传入的参数里定义的handle的执行操作。

那能不能做到自身处理完以后,然后在让继任者继续处理呢?答案是肯定的,但是在调用的handle以后,需要利用原型的特性调用如下代码:

Handler.prototype.handle.call(this);

该句话的意思说,调用原型的handle方法,来继续调用其继任者(也就是successor )的handle方法,以下代码表现为:button/dialog/app三个对象定义的handle都会执行。
var app = new Handler({

    handle: function () {

        console.log('app handle');

    }

}, 3);
var dialog = new Handler(app, 1);

dialog.handle = function () {

    console.log('dialog before ...')

    // 这里做具体的处理操作

    Handler.prototype.handle.call(this); //继续往上走

    console.log('dialog after ...')

};
var button = new Handler(dialog, 2);

button.handle = function () {

    console.log('button before ...')

    // 这里做具体的处理操作

    Handler.prototype.handle.call(this);

    console.log('button after ...')

};
button.handle();

通过代码的运行结果我们可以看出,如果想先自身处理,然后再调用继任者处理的话,就在末尾执行Handler.prototype.handle.call(this);代码,如果想先处理继任者的代码,就在开头执行Handler.prototype.handle.call(this);代码。

总结

职责链模式经常和组合模式一起使用,这样一个构件的父构件可以作为其继任者。

同时,DOM里的事件冒泡机制也和此好像有点类似,比如点击一个按钮以后,如果不阻止冒泡,其click事件将一直向父元素冒泡,利用这个机制也可以处理很多相关的问题,比如本系列设计模式享元模式里的《例1:事件集中管理》的示例代码。

Javascript 相关文章推荐
javascript学习之闭包分析
Dec 02 Javascript
在线所见即所得HTML编辑器的实现原理浅析
Apr 25 Javascript
JS+CSS实现自适应选项卡宽度的圆角滑动门效果
Sep 15 Javascript
jQuery打字效果实现方法(附demo源码下载)
Dec 18 Javascript
javascript HTML5 Canvas实现圆盘抽奖功能
Apr 11 Javascript
jQuery插件FusionCharts绘制的3D饼状图效果实例【附demo源码下载】
Mar 03 Javascript
vue与bootstrap实现时间选择器的示例代码
Aug 26 Javascript
vue基于Element构建自定义树的示例代码
Sep 19 Javascript
Vue.js通用应用框架-Nuxt.js的上手教程
Dec 25 Javascript
详解关于Vue2.0路由开启keep-alive时需要注意的地方
Sep 18 Javascript
在webstorm开发微信小程序之使用阿里自定义字体图标的方法
Nov 15 Javascript
JavaScript DOM常用操作代码汇总
Jul 03 Javascript
教你如何使用firebug调试功能了解javascript闭包和this
Mar 04 #Javascript
深入理解JavaScript系列(37):设计模式之享元模式详解
Mar 04 #Javascript
jQuery插件开发的五种形态小结
Mar 04 #Javascript
深入理解JavaScript系列(36):设计模式之中介者模式详解
Mar 04 #Javascript
百度UEditor编辑器如何关闭抓取远程图片功能
Mar 03 #Javascript
jQuery实现复选框成对选择及对应取消的方法
Mar 03 #Javascript
js实现文本框中输入文字页面中div层同步获取文本框内容的方法
Mar 03 #Javascript
You might like
十天学会php之第二天
2006/10/09 PHP
PHP4和PHP5性能测试和对比 测试代码与环境
2007/08/17 PHP
php使用array_rand()函数从数组中随机选择一个或多个元素
2014/04/28 PHP
JavaScript的parseInt 进制问题
2009/05/07 Javascript
Javascript 汉字字节判断
2009/08/01 Javascript
按下回车键指向下一个位置的一个函数代码
2014/03/10 Javascript
JS实现仿新浪微博发布内容为空时提示功能代码
2015/08/19 Javascript
完美实现八种js焦点轮播图(上篇)
2016/07/18 Javascript
Bootstrap基本组件学习笔记之分页(12)
2016/12/08 Javascript
基于jQuery实现火焰灯效果导航菜单
2017/01/04 Javascript
函数四种调用模式以及其中的this指向
2017/01/16 Javascript
js放大镜放大购物图片效果
2017/01/18 Javascript
bootstrapValidator.min.js表单验证插件
2017/02/09 Javascript
原JS实现banner图的常用功能
2017/06/12 Javascript
JS实现给json数组动态赋值的方法示例
2020/03/19 Javascript
基于rem的移动端响应式适配方案(详解)
2017/07/07 Javascript
史上最全JavaScript常用的简写技巧(推荐)
2017/08/17 Javascript
Vue.directive 自定义指令的问题小结
2018/03/04 Javascript
详解vue-cli 3.0 build包太大导致首屏过长的解决方案
2018/11/10 Javascript
Vue + Node.js + MongoDB图片上传组件实现图片预览和删除功能详解
2020/04/29 Javascript
在vue中封装方法以及多处引用该方法详解
2020/08/14 Javascript
Python functools模块学习总结
2015/05/09 Python
Linux上安装Python的PIL和Pillow库处理图片的实例教程
2016/06/23 Python
Python操作RabbitMQ服务器实现消息队列的路由功能
2016/06/29 Python
Python爬虫实现全国失信被执行人名单查询功能示例
2018/05/03 Python
python 用户交互输入input的4种用法详解
2019/09/24 Python
python3中rank函数的用法
2019/11/27 Python
从pandas一个单元格的字符串中提取字符串方式
2019/12/17 Python
利用CSS3实现文字折纸效果实例代码
2018/07/10 HTML / CSS
香港太阳眼镜网上商店:SmartBuyGlasses香港
2016/07/22 全球购物
武汉英思工程科技有限公司–ORACLE面试测试题目
2012/04/30 面试题
有模特经验的简历自我评价
2013/09/19 职场文书
家长对老师的感言
2014/03/11 职场文书
工地标语大全
2014/06/18 职场文书
2014年转正工作总结
2014/11/08 职场文书
博士给导师的自荐信
2015/03/06 职场文书