ホームページ > ウェブフロントエンド > jsチュートリアル > JavaScript 責任連鎖パターンの概要

JavaScript 責任連鎖パターンの概要

一个新手
リリース: 2017-10-23 09:59:48
オリジナル
1496 人が閲覧しました

はじめに

責任連鎖パターンでは、複数のオブジェクトがリクエストを処理する機会を持つことができるため、リクエストの送信者と受信者の結合関係が回避されます。オブジェクトをチェーンに接続し、オブジェクトが処理するまでチェーンに沿ってリクエストを渡します。

リクエストが行われた後、最初のオブジェクトから開始して、リクエストを受信したチェーン内のオブジェクトは、リクエストを自ら処理するか、チェーン内の次の候補に転送します。リクエストを送信するオブジェクトは、どのオブジェクトがリクエストを処理するのかを知りません。つまり、リクエストには暗黙的な受信者が存在します。実行時に、どの候補も対応するリクエストに応答できます。候補の数は、実行時にどの候補がチェーンに参加するかを決定することもできます。

図は次のとおりです:

テキスト

(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) book クラスを定義します

var Book=function (bNm,bName,bAuthor,bType) {
    this.bNm=bNm;
    this.bName=bName;
    this.bAuthor=bAuthor;
    this.bType=bType;
}
ログイン後にコピー

(4) Bookshelf クラス = 本棚 + 書籍

#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;
    }
ログイン後にコピー

なお、配列のフィルターメソッドの展開コードは以下の通りです

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.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 責任連鎖パターンの概要の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

関連ラベル:
ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新の問題
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート