论坛首页 Java企业应用论坛

单例/单体模式(Singleton)

浏览 10333 次
精华帖 (0) :: 良好帖 (1) :: 新手帖 (11) :: 隐藏帖 (2)
作者 正文
   发表时间:2010-05-17   最后修改:2011-04-07

首先,单例模式是作为对象的创建模式,此外还包括工厂模式。单例模式的三个特点:
1,该类只有一个实例
2,该类自行创建该实例(在该类内部创建自身的实例对象)
3,向整个系统公开这个实例接口

java中大概是这个样子:

 

class Singleton {
	
	//私有,静态的类自身实例
	private static Singleton instance = new Singleton();
	
	//私有的构造子(构造器,构造函数,构造方法)
	private Singleton(){}
	
	//公开,静态的工厂方法
	public static Singleton getInstance() {
		return instance;
	}
}

 

使用时:
Singleton obj = Singleton.getInstance();

 

这个单例类在自身被加载时instance会被实例化,即便加载器是静态的。
因此,对于资源密集,配置开销较大的单体更合理的做法是将实例化(new)推迟到使用它的时候。
即惰性加载(Lazy loading),它常用于那些必须加载大量数据的单体。修改下:

 

class LazySingleton {
	//初始为null,暂不实例化
	private static LazySingleton instance = null;
	
	//私有的构造子(构造器,构造函数,构造方法)
	private LazySingleton(){}
	
	//公开,静态的工厂方法,需要使用时才去创建该单体
	public static LazySingleton getInstance() {
		if( instance == null ) {
			instance = new LazySingleton();
		}
		return instance;
	}	
}
 

使用方式同上。

 

单例模式是javascript最基本,最有用的模式之一。它提供了一种将代码组织为一个逻辑单元的手段,这个逻辑单元中的代码通过单一的变量进行访问。
单体在javascipt中有许多用处,可以用来划分命名空间,以减少全局变量的泛滥。还可以用在分支技术中用来处理各浏览器的差异。
javascript中单例模式的实现方式有多种,每一种都有自身的优点或缺点:

 

1,对象直接量实现最基本,最简单的单体

 

var Singleton = {
	attr1 : 1,
	attr2 : 'hello',
	method1 : function(){alert(this.attr2);},
	method2 : function(arg){}
}

 

这种方式中,对象所有成员都通过Singleton加点号访问。所有成员是公开的,没有私有的。
在执行到变量Singleton时,会加载(实例化)自身,即非惰性加载。
此外method1用this访问单体的其它成员会存在一些风险,因为method1的上下文不是总是指向Singleton对象。
比如当把method1作为事件监听器时,this可能指向的是dom元素,这时可能会提示undefined。

 

 

2,闭包实现私有成员的单体

 

var Singleton = function(){
	var attr = 1, fn = function(){};
	return {
		method : function(){ fn(); },
		getAttr : function(){ return attr; }		
	};	
}();

 

这种方式中var定义私有的成员属性attr,方法fn,然后返回一个公开的接口method和getAttr。
今后修改实现时,接口方法method和getAttr不变,只需修改私有的attr和fn的具体实现。
使用如下:

 

Singleton.method();
Singleton.getAttr();

 

3,闭包实现私有成员的惰性实例化单体

 

var LazySingleton = function(){
	var attr = 1, fn = function(){};
	var obj = {
		method : function(){ fn(); },
		getAttr : function(){ return attr; }
	};
	function init(){
		return obj;
	}
	return {getInstace: init};
}();
 

适用场合上面已经提到:对于那些必须加载大量数据的单体直到需要使用它的时候才实例化。使用方式是这样的:


LazySingleton.getInstance().method();
LazySingleton.getInstance().getAttr();

 

 

 

附件:js单例模式实现事件管理模块的各个版本练习。

 

 

   发表时间:2010-05-18  
应该考虑到多线程环境下,还有序列化情况下。
0 请登录后投票
   发表时间:2010-05-18  
楼上的看过effective java吧
0 请登录后投票
   发表时间:2010-05-18  
class LazySingleton {
//初始为null,暂不实例化
private static Singleton instance = null;

//私有的构造子(构造器,构造函数,构造方法)
private Singleton(){}

//公开,静态的工厂方法,需要使用时才去创建该单体
public static Singleton getInstance() {
if( instance == null ) {
instance = new LazySingleton();
}
return instance;
}
}
唉,看不懂
0 请登录后投票
   发表时间:2010-05-18   最后修改:2010-05-18
class LazySingleton {
	//初始为null,暂不实例化
	private static Singleton instance = null;
	
	//私有的构造子(构造器,构造函数,构造方法)
	private Singleton(){}
	
	//公开,静态的工厂方法,需要使用时才去创建该单体
	public static Singleton getInstance() {
		if( instance == null ) {
			instance = new LazySingleton();
		}
		return instance;
	}	
}


理论上这里可能getInstance()获得的对象是非线程安全的
0 请登录后投票
   发表时间:2010-05-18  
多线程情况下使用双锁机制,就解决了。
0 请登录后投票
   发表时间:2010-05-19  
解决多线程的问题有3种方案:
1,直接用同步getInstance()方法,该方法不考虑性能的开销
2,早实例(early instantiate),即相对于懒实例。
3.双锁机制。前提:至少jdk1.5
0 请登录后投票
   发表时间:2010-05-22  
真的不错,现在做项目自己编写的这方面的代码很少,好好学习学习
0 请登录后投票
   发表时间:2010-05-23  
Lazy loading singleton 一般实现采用:
public static Singleton getInstance() {
  if (instance == null) {
    synchronized(Singleton.class) {
      if (instance == null) instance = new Singleton();
    }
  }
  return instance;
}


还有一种最高效的实现,由java设计者写出来的
public class Singleton {

  private static final class A {
    public static final Singleton s = new Singleton();
  }
  
  private Singleton() {
    
  }
  
  public static Singleton getInstance() {
    return A.s;
  }
}
0 请登录后投票
   发表时间:2010-06-23  
我记得这样的方式,如果使用反射,访问控制会无效
0 请登录后投票
论坛首页 Java企业应用版

跳转论坛:
Global site tag (gtag.js) - Google Analytics