JavaScript設(shè)計模式之單例模式原理與用法實例分析
本文實例講述了JavaScript設(shè)計模式之單例模式原理與用法。分享給大家供大家參考,具體如下:
單例模式的定義:保證一個類僅有一個實例,并提供一個訪問它的全局訪問點(diǎn)。
單例模式是一種常用的模式,有些對象只需要一個,如線程池、全局緩存、瀏覽器中的window對象等,這時候可以用到單例模式。
單例模式典型的應(yīng)用場景:單擊按鈕時,頁面中會出現(xiàn)一個登陸浮窗,而該登錄浮窗是唯一的,無論單擊多少次按鈕,這個浮窗都會被創(chuàng)建一次,則適合用單例模式創(chuàng)建。
全局變量不是單例模式,但在JavaScript開發(fā)中,經(jīng)常會把全局變量當(dāng)成單例來使用。
使用var a = {};這種方式創(chuàng)建對象a時,對象a是獨(dú)一無二的,若變量a被聲明在全局作用域下,則可以在代碼的任何位置使用這個變量。這顯然滿足單例模式的兩個條件。
但是全局變量存在很多問題,很容易造成命名空間污染,如上面的var a = {};隨時有可能被覆蓋。
有必要盡量減少全局變量的使用,即使需要,也應(yīng)該把它的污染降到最低。
降低全局變量帶來的命名污染的幾種方式:
1) 使用命名空間
適當(dāng)?shù)厥褂妹臻g,并不會杜絕全局變量,但可以減少全局變量的數(shù)量。
把a(bǔ)和b都定義為namespace的屬性,這樣可以減少變量和全局作用域打交道的機(jī)會:
var namespace = {
a: function() {
alert("a");
},
b: function() {
alert("b");
}
};
動態(tài)地創(chuàng)建命名空間:
var obj = {};
obj.namespace = function(name) {
var tips = name.split('.');
var cur = obj;
for (var i in tips) {
if (!cur[tips[i]])
cur[tips[i]] = {};
cur = cur[tips[i]];
}
};
obj.namespace('name');
obj.namespace('birth.year');
console.dir(obj);
上述代碼等價于:
var obj = {
name: {},
birth: {
year: {}
}
};
2) 使用閉包封裝私有變量
var person = (function() {
var_name = "Alice";
var _id = 16;
return {
getUserInfo: function() {
return _name + ": " + _id;
}
}
})();
使用下劃線來約定私有變量_name和_age,它們被封裝在閉包產(chǎn)生的作用域中,外部是訪問不到這兩個變量的,這就避免了對全局的命令污染。
惰性單例模式:
在需要的時候才創(chuàng)建對象實例。
var createBox = (function() {
var div;
return function() {
if (!div) {
div = document.createElement('div');
div.innerHTML = '登錄';
div.style.display = 'none';
document.body.appendChild(div);
}
return div;
}
})();
document.getElementById('btn').onclick = function() {
var box = createBox();
box.style.display = 'block';
};
用變量div來判斷是否已經(jīng)創(chuàng)建過浮窗。
通用的惰性單例:
問題:上面的惰性單例實例是違反單一職責(zé)原則的,創(chuàng)建對象和管理單例的邏輯都放在createBox對象內(nèi)部。若下次要創(chuàng)建頁面中唯一的iframe,需要把createBox幾乎照抄一遍。
var createIframe = (function() {
var iframe;
return function() {
if (!iframe) {
iframe = document.createElement('iframe');
document.body.appendChild(iframe);
return iframe;
}
return div;
}
})();
解決:把不變的部分隔離出來,其實,管理單例的邏輯可以完全抽象出來,因為它們的邏輯是一樣的:用一個變量來標(biāo)記是否創(chuàng)建過對象,若是,則在下次直接返回已經(jīng)創(chuàng)建好的對象。
var createSingle = function(func) {
var flag;
return flag || (flag = func.apply(this, arguments));
};
var createBox = function() {
var div = document.createElement('div');
div.innerHTML = '登錄';
div.style.display = 'none';
document.body.appendChild(div);
return div;
};
document.getElementById('btn').onclick = function() {
var box = createBox();
box.style.display = 'block';
};
var createIframe = createSingle(function() {
var iframe = document.createElement('iframe');
document.body.appendChild(iframe);
return iframe;
});
document.getElementById('btn').onclick = function() {
var iframe = createIframe();
iframe.style.display = 'block';
};
單例模式的其他用途:
單例模式的用途遠(yuǎn)不止于創(chuàng)建對象,比如click事件只需要在第一次渲染頁面時綁定一次,顯然運(yùn)用jQuery的one()方法可以實現(xiàn)。若運(yùn)用createSingle 方法,也很容易實現(xiàn):
var createSingle = function(func) {
var flag;
return flag || (flag = func.apply(this, arguments));
};
var bindEvent = createSingle(function() {
document.getElementById(‘div').onclick = function() {
...
};
return true;
});
var render = function() {
bindEvent();
};
render();
render();
render();
render()函數(shù)與bindEvent()函數(shù)執(zhí)行了3次,但div實際上只被綁定了一次。
更多關(guān)于JavaScript相關(guān)內(nèi)容可查看本站專題:《javascript面向?qū)ο笕腴T教程》、《JavaScript切換特效與技巧總結(jié)》、《JavaScript查找算法技巧總結(jié)》、《JavaScript錯誤與調(diào)試技巧總結(jié)》、《JavaScript數(shù)據(jù)結(jié)構(gòu)與算法技巧總結(jié)》、《JavaScript遍歷算法與技巧總結(jié)》及《JavaScript數(shù)學(xué)運(yùn)算用法總結(jié)》
希望本文所述對大家JavaScript程序設(shè)計有所幫助。
- JavaScript設(shè)計模式---單例模式詳解【四種基本形式】
- JS 設(shè)計模式之:單例模式定義與實現(xiàn)方法淺析
- javascript設(shè)計模式 – 單例模式原理與應(yīng)用實例分析
- 《javascript設(shè)計模式》學(xué)習(xí)筆記三:Javascript面向?qū)ο蟪绦蛟O(shè)計單例模式原理與實現(xiàn)方法分析
- js設(shè)計模式之單例模式原理與用法詳解
- JavaScript設(shè)計模式之單例模式簡單實例教程
- JS基于設(shè)計模式中的單例模式(Singleton)實現(xiàn)封裝對數(shù)據(jù)增刪改查功能
- JS設(shè)計模式之單例模式(一)
- NodeJS設(shè)計模式總結(jié)【單例模式,適配器模式,裝飾模式,觀察者模式】
- 學(xué)習(xí)JavaScript設(shè)計模式之單例模式
- JavaScript設(shè)計模式之單例模式實例
- JavaScript中的設(shè)計模式 單例模式
相關(guān)文章
JS非Alert實現(xiàn)網(wǎng)頁右下角“未讀信息”效果彈窗
這篇文章主要介紹了JS非Alert實現(xiàn)網(wǎng)頁右下角“未讀信息”效果彈窗的方法,涉及JavaScript動態(tài)操作頁面元素實現(xiàn)彈窗效果的相關(guān)技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-09-09
類似php的js數(shù)組的in_array函數(shù)自定義方法
PHP的數(shù)組函數(shù)in_array()非常方便,下面就為大家介紹下自定義類似php的js數(shù)組的in_array函數(shù),具體實現(xiàn)方法如下,感興趣的朋友可以參考下2013-12-12
JS 實現(xiàn)導(dǎo)航菜單中的二級下拉菜單的幾種方式
這篇文章主要介紹了JS 實現(xiàn)導(dǎo)航菜單中的二級下拉菜單的幾種方式的相關(guān)資料,這里提供了三種方式,和實例代碼,需要的朋友可以參考下2016-10-10

