JavaScript设计模式之责任链模式实例分析


Posted in Javascript onJanuary 16, 2019

本文实例讲述了JavaScript设计模式之责任链模式。分享给大家供大家参考,具体如下:

介绍

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

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

图解为:

JavaScript设计模式之责任链模式实例分析

正文

(1)由于类一般是与接口打交道的,为此我们先定义一个规范类中方法的接口,代码为

//定义一个静态方法来实现接口与实现类的直接检验
//静态方法不要写出Interface.prototype ,因为这是写到接口的原型链上的
//我们要把静态的函数直接写到类层次上
//定义一个接口类
var Interface=function (name,methods) {//name:接口名字
  if(arguments.length<2){
    alert("必须是两个参数")
  }
  this.name=name;
  this.methods=[];//定义一个空数组装载函数名
  for(var i=0;i<methods.length;i++){
    if(typeof methods[i]!="string"){
      alert("函数名必须是字符串类型");
    }else {
      this.methods.push( methods[i]);
    }
  }
};
Interface.ensureImplement=function (object) {
  if(arguments.length<2){
    throw new Error("参数必须不少于2个")
    return false;
  }
  for(var i=1;i<arguments.length;i++){
    var inter=arguments[i];
    //如果是接口就必须是Interface类型
    if(inter.constructor!=Interface){
      throw new Error("如果是接口类的话,就必须是Interface类型");
    }
    //判断接口中的方法是否全部实现
    //遍历函数集合分析
    for(var j=0;j<inter.methods.length;j++){
      var method=inter.methods[j];//接口中所有函数
      //object[method]传入的函数
      //最终是判断传入的函数是否与接口中所用函数匹配
      if(!object[method]||typeof object[method]!="function" ){//实现类中必须有方法名字与接口中所用方法名相同
        throw new Error("实现类中没有完全实现接口中的所有方法")
      }
    }
  }
}

(2)使用定义一个书店的接口

var bookShop=new Interface("bookShop",["addBook","findBook","showBooks"]);//书店接口

(3)定义一个书类

var Book=function (bNm,bName,bAuthor,bType) {
  this.bNm=bNm;
  this.bName=bName;
  this.bAuthor=bAuthor;
  this.bType=bType;
}

(4)书店类=书架+图书

1:在书店中添加书架和图书

var pcatBookShop=(function(){
 //书架
  var jsBooks = new Array();//js书架
  var cBooks = new Array();//c书架
  var javaBooks = new Array();//java书架
   //内部类1
  function AddJsBooks(book) {
    if(book.bType=="Js"){
      jsBooks.push(book);
    }else {
      AddJsBooks.successor(book);
    }
  }
  //内部类2
  function AddJavaBooks(book) {
    if(book.bType=="Java"){
      javaBooks.push(book);
    }else {
      AddJavaBooks.successor(book);
    }
  }
  //内部类3
  function AddCBooks(book) {
    if(book.bType=="C"){
      cBooks.push(book);
    }else {
      AddCBooks.successor(book);
    }
  }
})()

2:扩展设置责任链的方法(扩展在windows上)

//扩展window属性
window.setSuccessor=function (after,before) {
  after.successor=before;//引用的执行
}

3:设置责任链,将每个对象链接起来

//设置责任链-----串起来
setSuccessor(AddJsBooks,AddJavaBooks);
setSuccessor(AddJavaBooks,AddCBooks);

(5)查询图书的方法:通过图书编号和图书图书名称

 

/**********查询书籍************/
  var bookList = null;
  function FindBbn(keyword) {
    //链的头部来初始化参数
    if(!bookList){
      bookList=jsBooks.concat(cBooks).concat(javaBooks);
      var book = new Array();
      book=bookList.filter(function (book) {//对booklist进行过滤,过滤的条件为匿名函数
        if(book.bName.indexOf(keyword)!=-1){
             return true;
        }else {
          return false;
        }
      });
      //我要进行链式查询
      return book.concat(FindBbn.successor(keyword));
    }
  };
  function FindByName(keyword,book){
    var book = book;
    book = bookList.filter(function(book){
      if(book.bName.indexOf(keyword) != -1){
        return true;
      }else{
        return false;
      }
    });
    return book;
  }

注意,数组的filter方法扩展代码如下

Function.prototype.method=function (name,fn) {
  this.prototype[name]=fn;
  return this;
}
if(!Array.prototype.filter){
  Array.method("filter",function (fn,thisObj) {
    var scope=thisObj||window;
    var a=[];
    for(var i=0;i<this.length;i++){
      if(!fn.call(scope,this[i],i,this));{
        continue;
      }
      a.push(this[i]);
    }
    //返回过滤好数据
    return a;
  })
}

(6)规划责任链

setSuccessor(FindBbn,FindByName);

(7)真正的书店类(实现接口的类)

return function () {
    this.addBook=function (book) {
      if(book instanceof Book){
        AddJsBooks(book);//因为我知道谁是链的入口
      }
    };
    this.findBook=function (keyword) {
      return FindBbn(keyword);//游泳规划的责任链可以从头到尾的查询若,FindBbn没有则到FindByName中查询
    }
    this.showBooks=function () {
      document.write("JS类图书"+jsBooks.toSource()+"<br>");
      document.write("Java类图书"+javaBooks.toSource()+"<br>");
      document.write("C类图书"+cBooks.toSource()+"<br>");
      //自动生产----------
      document.write(cpoyStr(60,"-")+"<br>");
    }
  }

注意,在window上扩展一个可以自动生成“---------------”的方法

//扩展一个可以自动生产-----的方法
window.cpoyStr=function (num,str) {
  var newStr="";
  for(var i=0;i<num;i++){
   newStr+=str;
  }
  return newStr;
};

(8)使用书店

1:添加书

var pb = new pcatBookShop();
pb.addBook(new Book("00101","JAVA","JIM","JAVA"));
pb.addBook(new Book("00201","C#","world","C"));
pb.addBook(new Book("00202","C++/C","Hello","C"));
pb.addBook(new Book("00301","JAVASCRIPT","Good","JS"));

2:对书架上的书进行操作-----展示

//展示
pb.showBooks();
document.write(pb.findBook("C").toSource())

为此我们基本上完成了对责任链模式的使用方式的基本学习。

更多关于JavaScript相关内容还可查看本站专题:《javascript面向对象入门教程》、《JavaScript错误与调试技巧总结》、《JavaScript数据结构与算法技巧总结》、《JavaScript遍历算法与技巧总结》及《JavaScript数学运算用法总结》

希望本文所述对大家JavaScript程序设计有所帮助。

Javascript 相关文章推荐
删除条目时弹出的确认对话框
Jun 05 Javascript
JavaScript编程中容易出BUG的几点小知识
Jan 31 Javascript
基于百度地图实现产品销售的单位位置查看功能设计与实现
Oct 21 Javascript
jQuery实现模拟flash头像裁切上传功能示例
Dec 11 Javascript
js设置文字颜色的方法示例
Dec 30 Javascript
详谈js使用in和hasOwnProperty获取对象属性的区别
Apr 25 Javascript
Angular2学习教程之组件中的DOM操作详解
May 28 Javascript
[js高手之路]图解javascript的原型(prototype)对象,原型链实例
Aug 28 Javascript
关于vue-router的那些事儿
May 23 Javascript
element 中 el-menu 组件的无限极循环思路代码详解
Apr 26 Javascript
vue集成一个支持图片缩放拖拽的富文本编辑器
Jan 29 Vue.js
TS 类型兼容教程示例详解
Sep 23 Javascript
JavaScript设计模式之代理模式实例分析
Jan 16 #Javascript
vue-cli中vue本地实现跨域调试接口
Jan 16 #Javascript
vue element动态渲染、移除表单并添加验证的实现
Jan 16 #Javascript
深入koa-bodyparser原理解析
Jan 16 #Javascript
jQuery实现的点击图片居中放大缩小功能示例
Jan 16 #jQuery
详解微信小程序之scroll-view的flex布局问题
Jan 16 #Javascript
vue开发环境配置跨域的方法步骤
Jan 16 #Javascript
You might like
PHP字符串处理的10个简单方法
2010/06/30 PHP
FireFox浏览器使用Javascript上传大文件
2013/10/30 PHP
完善CodeIgniter在IDE中代码提示功能的方法
2014/07/19 PHP
qq登录,新浪微博登录接口申请过程中遇到的问题
2014/07/22 PHP
PHP按指定键值对二维数组进行排序的方法
2015/12/22 PHP
KindEditor在php环境下上传图片功能集成的方法示例
2020/07/20 PHP
A标签触发onclick事件而不跳转的多种解决方法
2013/06/27 Javascript
javascript读写XML实现广告轮换(兼容IE、FF)
2013/08/09 Javascript
深入理解Javascript中this的作用域
2014/08/12 Javascript
jquery比较简洁的软键盘特效实现方法
2015/03/19 Javascript
使用ngView配合AngularJS应用实现动画效果的方法
2015/06/19 Javascript
原生js实现数字字母混合验证码的简单实例
2015/12/10 Javascript
全面解析Bootstrap中nav、collapse的使用方法
2016/05/22 Javascript
jQuery控制元素隐藏和显示
2017/03/03 Javascript
JavaScript简单实现合并两个Json对象的方法示例
2017/10/16 Javascript
Vue组件之自定义事件的功能图解
2018/02/01 Javascript
layui加载数据显示loading加载完成loading消失的实例代码
2019/09/23 Javascript
vue弹出框组件封装实例代码
2019/10/31 Javascript
详解jQuery中的prop()使用方法
2020/01/05 jQuery
[03:03]DOTA2校园争霸赛 济南城市决赛欢乐发奖活动
2013/10/21 DOTA
[50:20]DOTA2上海特级锦标赛主赛事日 - 5 总决赛Liquid VS Secret第四局
2016/03/06 DOTA
在Django的视图中使用数据库查询的方法
2015/07/16 Python
python利用Guetzli批量压缩图片
2017/03/23 Python
python读取二进制mnist实例详解
2017/05/31 Python
Django中Forms的使用代码解析
2018/02/10 Python
对numpy中的transpose和swapaxes函数详解
2018/08/02 Python
python pandas.DataFrame.loc函数使用详解
2020/03/26 Python
基于python实现检索标记敏感词并输出
2020/05/07 Python
html5响应式开发自动计算fontSize的方法
2020/01/13 HTML / CSS
泰国的头号网上婴儿用品店:Motherhood.co.th
2019/04/09 全球购物
西安启天科技有限公司网络工程师面试题笔试题
2016/06/12 面试题
读书活动总结
2014/04/28 职场文书
暑假安全教育广播稿
2014/09/10 职场文书
毕业证丢失证明范本
2014/09/20 职场文书
区政府领导班子个人对照检查材料
2014/09/25 职场文书
Python函数中的不定长参数相关知识总结
2021/06/24 Python