<dfn id="w48us"></dfn><ul id="w48us"></ul>
  • <ul id="w48us"></ul>
  • <del id="w48us"></del>
    <ul id="w48us"></ul>
  • javascript 單例模式詳解及簡單實例

    時間:2024-08-17 11:52:30 JavaScript 我要投稿
    • 相關推薦

    javascript 單例模式詳解及簡單實例

      在Javascript中,單例模式是一種最基本又經常用到的設計模式,可能在不經意間就用到了單例模式。

      本文將從最基礎的理論開始,講述單例模式的基本概念和實現,最后用一個例子來講述單例模式的應用。

      理論基礎

      概念

      單例模式,顧名思義就是只有一個實例存在。通過單例模式可以保證系統中一個類只有一個實例而且該實例易于外界訪問,從而方便對實例個數的控制并節約系統資源。如果希望在系統中某個類的對象只能存在一個,單例模式是最好的解決方案。

      基本結構

      最簡單的單例模式起始就是一個對象字面量,它將有關聯的屬性和方法組織到一起。

      var singleton = { prop:"value", method:function(){ }}

      這種形式的單例模式,所有成員都是公開的,都可以通過singleton來訪問。這樣的缺點是單例中有一些輔助的方法并不希望暴露給使用者,如果使用者用了這些方法,然后在后面維護的時候,一些輔助方法被刪除,這樣會造成程序錯誤。

      如何避免這樣從的錯誤呢?

      包含私有成員的單例模式

      要怎么在類中創建私有成員呢,這通過需要閉包來進行實現,關于閉包的知識,本文不再贅述,大家可以自行Google。

      基本形式如下:

      var singleton = (function () { var privateVar = "private"; return { prop: "value", method: function () { console.log(privateVar); } } })();

      首先是一個自執行的匿名函數,在匿名函數中,聲明了一個變量privateVar,返回一個對象賦值給單例對象singleton。在匿名函數外部無法訪問到privateVar變量,它就是單例對象的私有變量,只能在函數內部或通過暴露出來的方法去訪問這個私有變量。這種形式又被成為模塊模式。

      惰性實例化

      不管是直接字面量或者私有成員的單例模式,兩者都是在腳本加載時就被創建出來的單例,但是有時候,頁面可能永遠也用不到這個單例對象,這樣會造成資源浪費。對于這種情況,最佳的處理方式就是惰性加載,就是說在需要的時候才去真正實例化這個單例對象,如何實現呢?

      var singleton = (function () { function init() { var privateVar = "private"; return { prop: "value", method: function () { console.log(privateVar); } } } var instance = null; return { getInstance: function () { if (!instance) { instance = init(); } return instance; } } })();

      首先將創建單例對象的代碼封裝到init函數中,然后聲明一個私有變量instance表示單例對象的實例,公開一個方法getInstance來獲取單例對象。

      調用的時候就通過singleton.getInstance()來進行調用,單例對象是在調用getInstance的時候才真正被創建。

      適用場合

      單例模式是JS中最常使用的設計模式,從增強模塊性和代碼組織性等方面來說,應該盡可能的使用單例模式。它可以把相關代碼組織到一起便于維護,對于大型項目,每個模塊惰性加載可以提高性能,隱藏實現細節,暴露出常用的api。常見的類庫比如underscore,jQuery我們都可以將其理解為單例模式的應用。

      結合實戰

      前面已經講過,單例模式是最常用的設計模式之一,我們來舉個例子進行說明,

      下面的代碼主要實現一個簡單的日期幫助類,通過單例模式實現:

      基本的單例模式結構

      var dateTimeHelper = { now: function () { return new Date(); }, format: function (date) { return date.getFullYear() + "-" + (date.getMonth() + 1) + "-" + date.getDate(); } }; console.log(dateTimeHelper.now());

      這段代碼通過對象字面量實現單例模式,使用的時候直接調用方法即可。

      惰性加載實現單例模式

      var dateTimeHelper = (function () { function init() { return { now: function () { return new Date(); }, format: function (date) { return date.getFullYear() + "-" + (date.getMonth() + 1) + "-" + date.getDate(); } } } var instance = null; return { getInstance: function () { if (!instance) { instance = init(); } return instance; } } })(); console.log(dateTimeHelper.getInstance().now())

      這就是惰性加載的單例模式。

      下面再來看幾個實例:

      實現1: 最簡單的對象字面量

      var singleton = { attr : 1, method : function(){ return this.attr; } }var t1 = singleton ;var t2 = singleton ;

      那么很顯然的, t1 === t2 。

      十分簡單,并且非常使用,不足之處在于沒有什么封裝性,所有的屬性方法都是暴露的。對于一些需要使用私有變量的情況就顯得心有余而力不足了。當然在對于 this 的問題上也是有一定弊端的。

      實現2:構造函數內部判斷

      其實和最初的JS實現有點類似,不過是將對是否已經存在該類的實例的判斷放入構造函數內部。

      function Construct(){ // 確保只有單例 if( Construct.unique !== undefined ){ return Construct.unique; } // 其他代碼 this.name = "NYF"; this.age="24"; Construct.unique = this;}var t1 = new Construct() ;var t2 = new Construct() ;

      那么也有的, t1 === t2 。

      也是非常簡單,無非就是提出一個屬性來做判斷,但是該方式也沒有安全性,一旦我在外部修改了Construct的unique屬性,那么單例模式也就被破壞了。

      實現3 : 閉包方式

      對于大著 靈活 牌子的JS來說,任何問題都能找到 n 種答案,只不過讓我自己去掂量孰優孰劣而已,下面就簡單的舉幾個使用閉包實現單例模式的方法,無非也就是將創建了的單例緩存而已。

      var single = (function(){ var unique; function Construct(){ // ... 生成單例的構造函數的代碼 } unique = new Constuct(); return unique;})();

      只要 每次講 var t1 = single; var t2 = single;即可。 與對象字面量方式類似。不過相對而言更安全一點,當然也不是絕對安全。

      如果希望會用調用 single() 方式來使用,那么也只需要將內部的 return 改為

      return function(){ return unique; }

      以上方式也可以使用 new 的方式來進行(形式主義的趕腳)。當然這邊只是給了閉包的一種例子而已,也可以在 Construct 中判斷單例是否存在 等等。 各種方式在各個不同情況做好選著即可。

      總結

      單例模式的好處在于對代碼的組織作用,將相關的屬性和方法封裝在一個不會被多次實例化的對象中,讓代碼的維護和調試更加輕松。隱藏了實現細節,可以防止被錯誤修改,還防止了全局命名空間的污染。另外可以通過惰性加載提高性能,減少不必要的內存消耗。

    【javascript 單例模式詳解及簡單實例】相關文章:

    JavaScript中push(),join() 函數實例詳解09-05

    常用的JavaScript模式09-22

    JavaScript fontcolor方法入門實例07-07

    javascript閉包的定義及應用實例分析08-25

    詳解JavaScript中的splice()使用方法08-20

    現代采購模式詳解12-02

    Javascript中arguments對象的詳解和使用方法08-20

    Java文件解壓縮實例詳解201607-26

    對javascript的理解08-08

    Javascript的this用法簡述08-15

    主站蜘蛛池模板: 久久久久亚洲精品中文字幕| 日韩精品无码一区二区三区免费| 欧美精品亚洲日韩aⅴ| 2018国产精华国产精品| 日韩精品欧美激情国产一区| 91精品成人免费国产片| 久久久久国产精品熟女影院| 麻豆精品| 国产精品H片在线播放| 2048亚洲精品国产| 中文字幕一精品亚洲无线一区| 成人午夜精品久久久久久久小说| 国内精品久久久久久99蜜桃| 亚洲精品国产精品乱码不卞| 国产亚洲精品a在线观看| 久久精品国产半推半就| 国产在线精品无码二区 | 日韩精品无码免费视频| 国产叼嘿久久精品久久| 精品福利资源在线| 国产成人无码精品一区二区三区| 久久亚洲中文字幕精品有坂深雪| 日本免费精品一区二区三区| 国内精品久久久久久久久电影网| 91精品福利在线观看| 亚洲国产精品久久| 久久亚洲国产欧洲精品一| 精品国产第1页| 欧美精品免费观看二区| 久久亚洲欧美日本精品| 欧美巨大黑人精品videos| 91麻豆精品国产自产在线观看一区 | 精品国产午夜肉伦伦影院| 亚洲午夜精品一区二区| 四虎国产精品免费入口| 国产精品中文字幕在线观看| 99国产精品久久| 久久伊人精品青青草原高清| 青青草原综合久久大伊人精品| 精品久久久久久亚洲| 精品一区二区三区中文字幕|