JavaScript設(shè)計(jì)模式之責(zé)任鏈模式實(shí)例分析
本文實(shí)例講述了JavaScript設(shè)計(jì)模式之責(zé)任鏈模式。分享給大家供大家參考,具體如下:
介紹
責(zé)任鏈模式(Chain of responsibility)是使多個對象都有機(jī)會處理請求,從而避免請求的發(fā)送者和接受者之間的耦合關(guān)系。將對象連成一條鏈,并沿著這條鏈傳遞該請求,直到有一個對象處理他為止。
請求以后,從第一個對象開始,鏈中收到請求的對象要么親自處理它,要么轉(zhuǎn)發(fā)給鏈中的下一個候選者。提交請求的對象并不知道哪一個對象將會處理它——也就是該請求有一個隱式的接受者(implicit receiver)。在運(yùn)行時,任一候選者都可以響應(yīng)相應(yīng)的請求,候選者的數(shù)目是任意的,也可以在運(yùn)行時刻決定哪些候選者參與到鏈中。
圖解為:

正文
(1)由于類一般是與接口打交道的,為此我們先定義一個規(guī)范類中方法的接口,代碼為
//定義一個靜態(tài)方法來實(shí)現(xiàn)接口與實(shí)現(xiàn)類的直接檢驗(yàn)
//靜態(tài)方法不要寫出Interface.prototype ,因?yàn)檫@是寫到接口的原型鏈上的
//我們要把靜態(tài)的函數(shù)直接寫到類層次上
//定義一個接口類
var Interface=function (name,methods) {//name:接口名字
if(arguments.length<2){
alert("必須是兩個參數(shù)")
}
this.name=name;
this.methods=[];//定義一個空數(shù)組裝載函數(shù)名
for(var i=0;i<methods.length;i++){
if(typeof methods[i]!="string"){
alert("函數(shù)名必須是字符串類型");
}else {
this.methods.push( methods[i]);
}
}
};
Interface.ensureImplement=function (object) {
if(arguments.length<2){
throw new Error("參數(shù)必須不少于2個")
return false;
}
for(var i=1;i<arguments.length;i++){
var inter=arguments[i];
//如果是接口就必須是Interface類型
if(inter.constructor!=Interface){
throw new Error("如果是接口類的話,就必須是Interface類型");
}
//判斷接口中的方法是否全部實(shí)現(xiàn)
//遍歷函數(shù)集合分析
for(var j=0;j<inter.methods.length;j++){
var method=inter.methods[j];//接口中所有函數(shù)
//object[method]傳入的函數(shù)
//最終是判斷傳入的函數(shù)是否與接口中所用函數(shù)匹配
if(!object[method]||typeof object[method]!="function" ){//實(shí)現(xiàn)類中必須有方法名字與接口中所用方法名相同
throw new Error("實(shí)現(xiàn)類中沒有完全實(shí)現(xiàn)接口中的所有方法")
}
}
}
}
(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書架
//內(nèi)部類1
function AddJsBooks(book) {
if(book.bType=="Js"){
jsBooks.push(book);
}else {
AddJsBooks.successor(book);
}
}
//內(nèi)部類2
function AddJavaBooks(book) {
if(book.bType=="Java"){
javaBooks.push(book);
}else {
AddJavaBooks.successor(book);
}
}
//內(nèi)部類3
function AddCBooks(book) {
if(book.bType=="C"){
cBooks.push(book);
}else {
AddCBooks.successor(book);
}
}
})()
2:擴(kuò)展設(shè)置責(zé)任鏈的方法(擴(kuò)展在windows上)
//擴(kuò)展window屬性
window.setSuccessor=function (after,before) {
after.successor=before;//引用的執(zhí)行
}
3:設(shè)置責(zé)任鏈,將每個對象鏈接起來
//設(shè)置責(zé)任鏈-----串起來 setSuccessor(AddJsBooks,AddJavaBooks); setSuccessor(AddJavaBooks,AddCBooks);
(5)查詢圖書的方法:通過圖書編號和圖書圖書名稱
/**********查詢書籍************/
var bookList = null;
function FindBbn(keyword) {
//鏈的頭部來初始化參數(shù)
if(!bookList){
bookList=jsBooks.concat(cBooks).concat(javaBooks);
var book = new Array();
book=bookList.filter(function (book) {//對booklist進(jìn)行過濾,過濾的條件為匿名函數(shù)
if(book.bName.indexOf(keyword)!=-1){
return true;
}else {
return false;
}
});
//我要進(jìn)行鏈?zhǔn)讲樵?
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;
}
注意,數(shù)組的filter方法擴(kuò)展代碼如下
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]);
}
//返回過濾好數(shù)據(jù)
return a;
})
}
(6)規(guī)劃責(zé)任鏈
setSuccessor(FindBbn,FindByName);
(7)真正的書店類(實(shí)現(xiàn)接口的類)
return function () {
this.addBook=function (book) {
if(book instanceof Book){
AddJsBooks(book);//因?yàn)槲抑勒l是鏈的入口
}
};
this.findBook=function (keyword) {
return FindBbn(keyword);//游泳規(guī)劃的責(zé)任鏈可以從頭到尾的查詢?nèi)?,F(xiàn)indBbn沒有則到FindByName中查詢
}
this.showBooks=function () {
document.write("JS類圖書"+jsBooks.toSource()+"<br>");
document.write("Java類圖書"+javaBooks.toSource()+"<br>");
document.write("C類圖書"+cBooks.toSource()+"<br>");
//自動生產(chǎn)----------
document.write(cpoyStr(60,"-")+"<br>");
}
}
注意,在window上擴(kuò)展一個可以自動生成“---------------”的方法
//擴(kuò)展一個可以自動生產(chǎn)-----的方法
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:對書架上的書進(jìn)行操作-----展示
//展示
pb.showBooks();
document.write(pb.findBook("C").toSource())
為此我們基本上完成了對責(zé)任鏈模式的使用方式的基本學(xué)習(xí)。
更多關(guān)于JavaScript相關(guān)內(nèi)容還可查看本站專題:《javascript面向?qū)ο笕腴T教程》、《JavaScript錯誤與調(diào)試技巧總結(jié)》、《JavaScript數(shù)據(jù)結(jié)構(gòu)與算法技巧總結(jié)》、《JavaScript遍歷算法與技巧總結(jié)》及《JavaScript數(shù)學(xué)運(yùn)算用法總結(jié)》
希望本文所述對大家JavaScript程序設(shè)計(jì)有所幫助。
相關(guān)文章
JavaScript?中如何攔截全局?Fetch?API?的請求和響應(yīng)問題
在本文中,我們介紹了什么是 JavaScript 攔截器,學(xué)習(xí)了如何通過給 Fetch API 使用猴子補(bǔ)丁和使用 fetch-intercept 庫來創(chuàng)建攔截器,對js攔截全局Fetch?API的請求和響應(yīng)知識感興趣的朋友跟隨小編一起看看吧2023-01-01
JS判斷數(shù)組是否包含某元素實(shí)現(xiàn)方法匯總
這篇文章主要介紹了JS判斷數(shù)組是否包含某元素實(shí)現(xiàn)方法匯總,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-06-06
基于JS實(shí)現(xiàn)頁面視頻video標(biāo)簽禁止下載(下載按鈕+右擊菜單)
最近做項(xiàng)目遇到這樣的需求,禁止用戶瀏覽頁面的時候下載頁面的視頻,網(wǎng)上看到下載視頻的方法有兩種,本文對每種方法做詳細(xì)分析,對js禁止下載視頻相關(guān)知識感興趣的朋友一起看看吧2024-02-02

