JavaScript代理模式

代理模式的定義,代理是一個對象(proxy)用它來控制目標對象的訪問。爲此他要是先與目標對象相同的接口,可是他不一樣於裝飾者模式,它對目標對象不進行任何修改,它的目的在於延緩"複雜"對象的初始化時間。這樣能夠在用到這個目標對象的時候再初始化他(對於單例來說更是重要)。數組

代理模式有兩種分類:函數

(1)普通代理this

(2)惰性代理spa

具體看下面的例子prototype

第一,普通代理模式代理

步驟一,接口檢驗文件的引用code

//定義一個靜態方法來實現接口與實現類的直接檢驗
//靜態方法不要寫出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)圖書類blog

  //圖書類
    /*
    * 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);
        }
    }

修改後的代理執行過程,如圖

相關文章
相關標籤/搜索