JavaScript代理模式原理與用法例項詳解
阿新 • • 發佈:2020-03-11
本文例項講述了JavaScript代理模式原理與用法。分享給大家供大家參考,具體如下:
代理模式的定義,代理是一個物件(proxy)用它來控制目標物件的訪問。為此他要是先與目標物件相同的介面,但是他不同於裝飾者模式,它對目標物件不進行任何修改,它的目的在於延緩"複雜"物件的初始化時間。這樣可以在用到這個目標物件的時候再初始化他(對於單例來講更是重要)。
代理模式有兩種分類:
(1)普通代理
(2)惰性代理
具體看下面的例子
第一,普通代理模式
步驟一,介面檢驗檔案的引用
//定義一個靜態方法來實現介面與實現類的直接檢驗 //靜態方法不要寫出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("實現類中沒有完全實現介面中的所有方法") } } } }
步驟二,目標類
(1)圖書類
//圖書類 /* * bid 圖書id * bName 圖書名稱 * bPrice 圖書價格 * */ var Book = function(bid,bName,bPrice){ this.bid = bid; this.bName = bName; this.bPrice = bPrice; }
(2)真正的目標類
//目標類 var myBookShop=(function () { //書店裡的書 var books={}; return function (bks) { //初始化 if(typeof bks=="object"){ books=bks; } //加書 this.addBook = function(book){ books[book.bid] = book; } //找書 this.findBook=function (bid) { if(books[bid]){ return books[bid]; }else { return null; } } //還書 this.returnBook=function (book) { this.addBook(book); } //借書 this.lendBook=function (bid) { var book=this.findBook(bid); return book; } } })();
步驟三,普通代理
var myBookShopProxy=function (bks) { var obj=new myBookShop(bks);//類似於目標類的引用 //加書 this.addBook=function (book) { obj.addBook(book); } //找書 this.findBook = function(bid){ return obj.findBook(bid); } //還書 this.returnBook=function (book) { obj.returnBook(book); } //借書 this.lendBook=function (bid) { return obj.lendBook(bid); } }
步驟四,新增資料後,開始訪問
var proxy = new myBookShopProxy({ "001":new Book("001","EXTJS","45"),"002":new Book("002","JS","60") }) alert(proxy.lendBook("001").bName)
在普通的代理模式中,我們可以看出代理中對目標物件的引用是一次性初始化的,然後再在該基礎上實現其他操作
如圖:目標類和代理同時實現了同一介面。代理中一次性對目標類進行例項,然後值訪問到目標類中的方法。
總結,這個代理是我們嚴格安裝定義來寫的,一般開發中不會用到,應為他沒什麼意義。
第二種,惰性代理----在使用時才對目標類進行初始化再引用。
對代理部分的修改如下,其餘部分相同,程式碼如下
//惰性代理---在使用時在初始化目標類的引用 var myBookShopProxy=function (bks) { var obj=null; this._init=function () { obj=new myBookShop(bks); } //加書 this.addBook=function (book) { this._init(); obj.addBook(book); } //找書 this.findBook=function (bid) { this._init(); obj.findBook(bid); } //還書 this.returnBook=function (book) { this._init(); obj.returnBook(book); } //借書 this.lendBook=function (bid) { this._init(); return obj.lendBook(bid); } }
修改後的代理執行過程,如圖
感興趣的朋友可以使用線上HTML/CSS/JavaScript前端程式碼除錯執行工具:http://tools.jb51.net/code/WebCodeRun測試上述程式碼執行效果。
更多關於JavaScript相關內容還可檢視本站專題:《javascript面向物件入門教程》、《JavaScript錯誤與除錯技巧總結》、《JavaScript資料結構與演算法技巧總結》、《JavaScript遍歷演算法與技巧總結》及《JavaScript數學運算用法總結》
希望本文所述對大家JavaScript程式設計有所幫助。