初学Java设计模式随记 -- 单例(Singleton)模式

本篇内容主要摘取自GoF的设计模式和阎宏的《Java与模式》 

 

单例(Singleton)模式,也叫单件模式。定义如下:(参照GoF设计模式)

 

1. 用意:

保证一个类仅有一个实例,并提供一个访问它的全局访问点。

 

2.参与者:
• 单例类(Singleton)
— 定义一个getInstance()操作,允许客户访问它的唯一实例。getInstance()是一个类操作(一个静态方法)。
— 可能负责创建它自己的唯一实例。

 

3.结构图:

 


根据这个定义,客户只能通过Singleton类的getInstance()方法操作访问一个Singleton的实例。 

 

在阎宏的《Java与模式》中,给出了下面的定义:

单例(Singleton)模式确保一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。这个类称为单例类。

 

什么时候用单例模式呢?

从定义就可以看出来,就是当系统要求一个类只有一个实例时,才应当使用单例模式。

 

单例模式的要点和实现:

1. 单例类只能有一个实例 -- 一个静态的单例类类型的成员变量;

2. 单例类只能自己创建自己的唯一实例 -- 构造函数应该是私有的;

3. 单例类必须给所有其他对象提供这一实例 -- 有一个共有的静态方法来返回单例类自己的实例。

 

 

Java中,单例模式有两种形态: 饿汉式(eager)懒汉式(lazy)。

对应的代码如下:

 

饿汉式(eager)

单例类在自己被加载时就将自己实例化。

  

/*
 * 饿汉(eager)式单例类
 */
public class EagerSingleton {
	
//	一个静态的单例类类型的成员变量
	private static final EagerSingleton m_instance = new EagerSingleton();
	
	/**
	 * 私有构造函数,保证外界无法直接实例化
	 */
	private EagerSingleton(){}
	
	/**
	 * 静态工厂方法,返回单例类自己的实例
	 */
	public static EagerSingleton getInstance(){
		return m_instance;
	}
}

 

懒汉式(lazy)

单例类在第一次被引用时将自己实例化。

懒汉式(lazy)必须处理好在多个线程同时首次引用此类时的访问限制问题。 

 

/*
 * 懒汉(lazy)式单例类
 */
public class LazySingleton {
	
//	一个静态的单例类类型的成员变量
	private static LazySingleton m_instance = null;
	
	/**
	 * 私有构造函数,保证外界无法直接实例化
	 */
	private LazySingleton(){}
	
	/**
	 * 静态工厂方法,返回单例类自己的实例,
	 * 使用了同步化synchronized,和一个if判断,来处理多线程环境。
	 */
	
	synchronized public static LazySingleton getInstance(){
		if(m_instance == null){
			m_instance = new LazySingleton();
		}
		
		return m_instance;
	}

}

 

单例模式还有一种形态使用了双重检查成例,不过在阎宏的《JAVA与模式》中,指出双重检查成例并不适合Java编译器。

基本原因在于,在Java编译器,LazySingleton类的初始化与m_instance的变量赋值的顺序不可预料。如果一个线程在没有同步化的条件下读取m_instance引用,并调用这个对象的方法的话,可能会发现对象的初始化过程尚未完成,从而造成崩溃。

 

双重检查成例不适合Java是由于Java的内存模型造成的,参照 双重检查锁定及单例模式

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值