JS設計模式---3.封裝

封裝之利

  • 保證內部數(shù)據(jù)完整浆熔,易于重構
  • 弱化模塊間耦合邢疙,提高對象可重用性

封裝之弊

  • 單元測試困難暖哨,錯誤調試困難
  • 過度封裝會損失類的靈活性
  • 對新手不友好

創(chuàng)建對象的基本模式

需求:創(chuàng)建一個存儲一本書的類目派,并為其實現(xiàn)一個以HTML形式顯示數(shù)據(jù)的方法然低。你只負責創(chuàng)建這個類仔役,別人會創(chuàng)建和使用其實例掷伙。

// 使用方式
var theHobbit = new Book('0-395-07122-4','The Hobbit','J.R.R. Tolkien');
theHobbit.display() // 以HTML形式顯示數(shù)據(jù)
門戶大開型對象
  var Publication = new interface('Publication', ['getIsbn', 'setIsbn', 'getTitle', 'setTitle', 'getAuthor',
    'setAuthor', 'display'
  ])
  var Book = function (isbn, title, author) {
    this.setIsbn(isbn);
    this.setTitle(title);
    this.setAuthor(author);
  }
  Book.prototype = {
    checkIsbn = function (isbn) {
      // ...
    },
    getIsbn: function () {
      return this.isbn;
    },
    setIsbn: function (isbn) {
      if (!this.checkIsbn(isbn)) throw new Error('Book: Invalid ISBN');
      this.isbn = isbn;
    },
    getTitle: function () {
      return this.title;
    },
    setTitle: function (title) {
      this.title = title || 'No title specified';
    },
    getAuthor: function () {
      return this.author;
    },
    setAuthor: function (author) {
      this.author = author || 'No author specified';
    },
    display: function () {
      //  ...
    }
  }

上述代碼定義了一個接口,那么其他程序員就應該只能使用接口中定義的屬性和方法又兵。
這是門戶大開型對象創(chuàng)建方式所能得到的最好結果任柜。一個明確定義的接口,一些數(shù)據(jù)的賦值器和取值器沛厨,及一些檢驗方法宙地。
缺點就是雖然我們定義了賦值器方法,但是這些屬性仍然是公開的逆皮、可以直接設置的宅粥。

命名規(guī)范區(qū)分私有成員
 var Book = function (isbn, title, author) {
    this.setIsbn(isbn);
    this.setTitle(title);
    this.setAuthor(author);
  }
  Book.prototype = {
    _checkIsbn = function (isbn) {
      // ...
    },
    getIsbn: function () {
      return this._isbn;
    },
    setIsbn: function (isbn) {
      if (!this.checkIsbn(isbn)) throw new Error('Book: Invalid ISBN');
      this._isbn = isbn;
    },
    getTitle: function () {
      return this._title;
    },
    setTitle: function (title) {
      this._title = title || 'No title specified';
    },
    getAuthor: function () {
      return this._author;
    },
    setAuthor: function (author) {
      this._author = author || 'No author specified';
    },
    display: function () {
      //  ...
    }
  }

這是一種程序員約定俗成的方法,加下劃線表示私有變量电谣。但是JS中跟本沒有私有變量的定義秽梅,只能說是總所周知的命名規(guī)范抹蚀。缺點也很明顯,既然是約定企垦,那么只有在遵守時才有效果环壤。

閉包實現(xiàn)私用成員
 var Book = function (newIsbn, newTitle, newAuthor) {
    //私有屬性
    var isbn, title, author;
    // 私有方法
    function checkIsbn(isbn) {
      // ...
    };
    // 特權方法
    this.getIsbn = function () {
      return isbn;
    };
    this.setIsbn = function (newIsbn) {
      if (!checkIsbn(newIsbn)) throw new Error('Book:Invalid ISBN');
      isbn = newIsbn;
    };
    this.getTitle = function () {
      return title;
    };
    this.setTitle = function (newTitle) {
      title = newTitle || 'NO title specified';
    };
    this.getAuthor = function () {
      return author;
    };
    this.setAuthor = function (newAuthor) {
      author = newAuthor || 'NO author specified';
    };
    //  構造函數(shù) 賦值
    this.setIsbn(newIsbn);
    this.setTitle(newTitle);
    this.setAuthor(newAuthor);
  };
 // 公共方法
  Book.prototype = {
    display() {
      // ...
    }
  }

這種方式創(chuàng)建的對象具有真正的私有屬性,解決了可以直接取值賦值的問題钞诡。缺點是會耗費更多的內存郑现,而且不利于派生子類。在JS中荧降,用閉包實現(xiàn)私用成員導致的派生問題被稱為”繼承破壞封裝“接箫。

MORE

靜態(tài)成員

作用域和閉包可用于創(chuàng)建靜態(tài)成員。大多數(shù)方法和屬性所關聯(lián)的是類的實例朵诫,而靜態(tài)成員所關聯(lián)的是類本身辛友。

  • 一個栗子
  var Book = (function () {
    // 私有靜態(tài)屬性
    var numOfBookes = 0;
    // 私有靜態(tài)方法
    function checkIsbn(isbn) {
      // ...
    };
    // 返回構造器
    return function (newIsbn, newTitle, newAuthor) {
      // 私有屬性
      var isbn, title, author;
      // 特權方法
      this.getIsbn = function () {
        return isbn;
      };
      this.setIsbn = function (newIsbn) {
        if (!checkIsbn(newIsbn)) throw new Error('Book:Invalid ISBN');
        isbn = newIsbn;
      };
      this.getTitle = function () {
        return title;
      };
      this.setTitle = function (newTitle) {
        title = newTitle || 'NO title specified';
      };
      this.getAuthor = function () {
        return author;
      };
      this.setAuthor = function (newAuthor) {
        author = newAuthor || 'NO author specified';
      };

      numOfBookes++;
      if (numOfBookes > 50) {
        throw new Error('Book:Only 50 instance of book can be created.')
      }

      // 賦值
      this.setIsbn(newIsbn);
      this.setTitle(newTitle);
      this.setAuthor(newAuthor);
    }
  })();
  // 公共靜態(tài)方法
  Book.converToTitleCase = function (inputString) {
    // ...
  };
  // 公共方法 非特權
  Book.prototype = {
    display:function() {
      // ...
    }
  }

閉包,返回一個構造器拗窃,私用成員和特權成員仍然在構造器中瞎领。但是閉包中可以訪問靜態(tài)成員。優(yōu)點在于所有的靜態(tài)成員只會存在一份随夸,這樣大大減小了內存的消耗九默。
q:如何判斷是否設計成靜態(tài)成員?
a:一般情況下宾毒,我們只需要看一個屬性或者方法是否需要訪問實例數(shù)據(jù)驼修。如果不需要,那么設計成靜態(tài)成員會更有效率诈铛。

常量

常量的取值

  var Class = (function () {
    var constans = {
      UPPER_BOUND: 100,
      LOWER_BOUND: -100
    };
    var ctor = function () {
      // ...
    }
    // 特權靜態(tài)方法
    ctor.getConstans = function (name) {
      return constans[name];
    }
    return ctor
  })()
  // 取值
  Class.getConstans('UPPER_BOUND')
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末乙各,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子幢竹,更是在濱河造成了極大的恐慌耳峦,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,651評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件焕毫,死亡現(xiàn)場離奇詭異蹲坷,居然都是意外死亡,警方通過查閱死者的電腦和手機邑飒,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,468評論 3 392
  • 文/潘曉璐 我一進店門循签,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人疙咸,你說我怎么就攤上這事县匠。” “怎么了?”我有些...
    開封第一講書人閱讀 162,931評論 0 353
  • 文/不壞的土叔 我叫張陵乞旦,是天一觀的道長贼穆。 經常有香客問我,道長杆查,這世上最難降的妖魔是什么扮惦? 我笑而不...
    開封第一講書人閱讀 58,218評論 1 292
  • 正文 為了忘掉前任臀蛛,我火速辦了婚禮亲桦,結果婚禮上,老公的妹妹穿的比我還像新娘浊仆。我一直安慰自己客峭,他們只是感情好,可當我...
    茶點故事閱讀 67,234評論 6 388
  • 文/花漫 我一把揭開白布抡柿。 她就那樣靜靜地躺著舔琅,像睡著了一般。 火紅的嫁衣襯著肌膚如雪洲劣。 梳的紋絲不亂的頭發(fā)上备蚓,一...
    開封第一講書人閱讀 51,198評論 1 299
  • 那天,我揣著相機與錄音囱稽,去河邊找鬼郊尝。 笑死,一個胖子當著我的面吹牛战惊,可吹牛的內容都是我干的流昏。 我是一名探鬼主播,決...
    沈念sama閱讀 40,084評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼吞获,長吁一口氣:“原來是場噩夢啊……” “哼况凉!你這毒婦竟也來了?” 一聲冷哼從身側響起各拷,我...
    開封第一講書人閱讀 38,926評論 0 274
  • 序言:老撾萬榮一對情侶失蹤刁绒,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后烤黍,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體知市,經...
    沈念sama閱讀 45,341評論 1 311
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,563評論 2 333
  • 正文 我和宋清朗相戀三年蚊荣,在試婚紗的時候發(fā)現(xiàn)自己被綠了初狰。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,731評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡互例,死狀恐怖奢入,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤腥光,帶...
    沈念sama閱讀 35,430評論 5 343
  • 正文 年R本政府宣布关顷,位于F島的核電站,受9級特大地震影響武福,放射性物質發(fā)生泄漏议双。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,036評論 3 326
  • 文/蒙蒙 一捉片、第九天 我趴在偏房一處隱蔽的房頂上張望平痰。 院中可真熱鬧,春花似錦伍纫、人聲如沸宗雇。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,676評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽赔蒲。三九已至,卻和暖如春良漱,著一層夾襖步出監(jiān)牢的瞬間舞虱,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,829評論 1 269
  • 我被黑心中介騙來泰國打工母市, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留矾兜,地道東北人。 一個月前我還...
    沈念sama閱讀 47,743評論 2 368
  • 正文 我出身青樓窒篱,卻偏偏與公主長得像焕刮,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子墙杯,可洞房花燭夜當晚...
    茶點故事閱讀 44,629評論 2 354

推薦閱讀更多精彩內容

  • Swift1> Swift和OC的區(qū)別1.1> Swift沒有地址/指針的概念1.2> 泛型1.3> 類型嚴謹 對...
    cosWriter閱讀 11,097評論 1 32
  • 設計模式概述 在學習面向對象七大設計原則時需要注意以下幾點:a) 高內聚配并、低耦合和單一職能的“沖突”實際上,這兩者...
    彥幀閱讀 3,741評論 0 14
  • 今天高镐,不知道多少人單方面失戀了溉旋,是的,宋仲基和宋慧喬公布婚訊了嫉髓」劾埃看到這條消息,我的CP粉特質也冒出來了算行,看...
    藍梓恩閱讀 196評論 0 0
  • 每天早起來梧油,都還能看到虛月掛與空中,呼吸著清晨最清新的空氣州邢,仿若自己以漂浮與大地之間儡陨。 這兩天陽光如此嬌嫩,以...
    歷練成圣閱讀 150評論 0 0