javascript实现单例模式附源码解析
日期:2019-03-29
来源:程序思维浏览:2522次
什么是单例模式?
单例模式也称作为单子模式,单体模式。单例模式的定义是产生一个类的唯一实例,是软件设计中较为简单但是很常用的一种设计模式。
单例模式的核心是确保只有一个实例,并提供全局访问。
实现代码如下:
var SingleTon = function(name){ //创建一个对象
this.name = name;
this.instance = null;
};
SingleTon.prototype.getName = function(){
alert(this.name);
};
SingleTon.getInstance = function(name){
if(!this.instance){
this.instance = new SingleTon(name);
}
return this.instance;
};
var a = SingleTon.getInstance( 'instance1' );
var b = SinleTon.getInstance( 'instance2' );
alert( a === b); //返回true
上述代码虽然实现了单例模式,但实际运用中并不这样运用,请看下面代码。
1.直接创建一个对象实现单例模式
既然我们只要一个实例,我们可以很容易想到,直接创建一个对象不就行了。
// 对象字面量
var Singleton = {
attr1: 1,
attr2: 2,
method1: function(){
return this.attr1;
},
method2: function(){
return this.attr2;
}
};
//对象变量
var div = document.body.appendChild( document.createElement( ''div' ) );
不足之处:
(1)没有什么封装性,所有的属性方法都是暴露的。
(2)全局变量很容易造成命名空间污染。
(3)对象一开始变创建,万一我们用不上就浪费了。
2.实现惰性单例(利用闭包和立即执行函数来实现)
// 实现单体模式创建div
var createDiv= (function(){
var div;
return function(){
if(!div) {
div = document.createElement("div");
div.style.width = '100px';
div.style.height = '100px';
div.style.background = '#e4e4e4';
document.body.appendChild(div);
}
return div;
}
})();
var div1=createDiv();
var div2=createDiv();
console.log(div1===div2); //true
虽然我们完成了惰性单例,但是我们同样发现了问题
违反了单一职责原则,创建对象和管理单例放在了一个函数中createDiv
如果我们还想创建一个其他的唯一对象,那就只能copy了
综上,我们需要 :
把不变的部分隔离出来,把可变的封装起来,这给予了我们扩展程序的能力,符合开放-封闭原则
<div id='test1' style="height:200px;background-color: #333"></div>
<div id='test2' style="height:200px;background-color: #e3e3e3"></div>
<script>
// 获取单个实例
var getInstance = function(fn) {
var result;
return function(){
return result || (result = fn.call(this,arguments));
}
};
// 创建div
var createWindow = function(){
var div = document.createElement("div");
div.innerHTML = "i am #div";
div.style.display = 'none';
document.body.appendChild(div);
return div;
};
// 创建iframe
var createIframe = function(){
var iframe = document.createElement("iframe");
document.body.appendChild(iframe);
return iframe;
};
// 测试创建div
var createSingleDiv = getInstance(createWindow); //返回一个闭包,result 为null
document.getElementById("test1").onclick = function(){
var win = createSingleDiv();
win.style.display = "block";
};
// 测试创建iframe
var createSingleIframe = getInstance(createIframe); //返回一个闭包,result 为null
document.getElementById("test2").onclick = function(){
var win = createSingleIframe();
win.src = "https://www.imooc.com";
};
</script>
源码下载
单例模式也称作为单子模式,单体模式。单例模式的定义是产生一个类的唯一实例,是软件设计中较为简单但是很常用的一种设计模式。
单例模式的核心是确保只有一个实例,并提供全局访问。
实现代码如下:
var SingleTon = function(name){ //创建一个对象
this.name = name;
this.instance = null;
};
SingleTon.prototype.getName = function(){
alert(this.name);
};
SingleTon.getInstance = function(name){
if(!this.instance){
this.instance = new SingleTon(name);
}
return this.instance;
};
var a = SingleTon.getInstance( 'instance1' );
var b = SinleTon.getInstance( 'instance2' );
alert( a === b); //返回true
上述代码虽然实现了单例模式,但实际运用中并不这样运用,请看下面代码。
1.直接创建一个对象实现单例模式
既然我们只要一个实例,我们可以很容易想到,直接创建一个对象不就行了。
// 对象字面量
var Singleton = {
attr1: 1,
attr2: 2,
method1: function(){
return this.attr1;
},
method2: function(){
return this.attr2;
}
};
//对象变量
var div = document.body.appendChild( document.createElement( ''div' ) );
不足之处:
(1)没有什么封装性,所有的属性方法都是暴露的。
(2)全局变量很容易造成命名空间污染。
(3)对象一开始变创建,万一我们用不上就浪费了。
2.实现惰性单例(利用闭包和立即执行函数来实现)
// 实现单体模式创建div
var createDiv= (function(){
var div;
return function(){
if(!div) {
div = document.createElement("div");
div.style.width = '100px';
div.style.height = '100px';
div.style.background = '#e4e4e4';
document.body.appendChild(div);
}
return div;
}
})();
var div1=createDiv();
var div2=createDiv();
console.log(div1===div2); //true
虽然我们完成了惰性单例,但是我们同样发现了问题
违反了单一职责原则,创建对象和管理单例放在了一个函数中createDiv
如果我们还想创建一个其他的唯一对象,那就只能copy了
综上,我们需要 :
把不变的部分隔离出来,把可变的封装起来,这给予了我们扩展程序的能力,符合开放-封闭原则
<div id='test1' style="height:200px;background-color: #333"></div>
<div id='test2' style="height:200px;background-color: #e3e3e3"></div>
<script>
// 获取单个实例
var getInstance = function(fn) {
var result;
return function(){
return result || (result = fn.call(this,arguments));
}
};
// 创建div
var createWindow = function(){
var div = document.createElement("div");
div.innerHTML = "i am #div";
div.style.display = 'none';
document.body.appendChild(div);
return div;
};
// 创建iframe
var createIframe = function(){
var iframe = document.createElement("iframe");
document.body.appendChild(iframe);
return iframe;
};
// 测试创建div
var createSingleDiv = getInstance(createWindow); //返回一个闭包,result 为null
document.getElementById("test1").onclick = function(){
var win = createSingleDiv();
win.style.display = "block";
};
// 测试创建iframe
var createSingleIframe = getInstance(createIframe); //返回一个闭包,result 为null
document.getElementById("test2").onclick = function(){
var win = createSingleIframe();
win.src = "https://www.imooc.com";
};
</script>
源码下载
精品好课