欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

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

程序员文章站 2023-11-24 12:23:46
本文实例讲述了javascript设计模式之责任链模式。分享给大家供大家参考,具体如下: 介绍 责任链模式(chain of responsibility)是使多个对象...

本文实例讲述了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程序设计有所帮助。