背景
保证只有一个实例时,你是否会想到单例模式
1、使用意图
保证只有一个实例,比如帮助窗体,没必要维护多个实例
2、生活实例
地球只有一个(不知道算不算单例的体现)
3、Java 例子(框架、JDK 、JEE)
单例模式的例子数不胜数,很常见的例子就是Spring的Controller是一个单例,但是如果按照单例模式编码规则来说,Servlet不是单例,但是他确实单实例,意思是Web容器只维护一个Servlet对象,其实也可以等同为单例的结果。另外,也不要一看到类名.getInstance()就以为那是单例模式,比如 Calendar.getInstance() 他不是单例模式,得到的对象都是不同是的实例。
/**
* Gets a calendar using the default time zone and locale. The
* Calendar
returned is based on the current time
* in the default time zone with the default locale.
*
* @return a Calendar.
*/
public static Calendar getInstance()
{
Calendar cal = createCalendar(TimeZone.getDefaultRef(), Locale.getDefault());
cal.sharedZone = true;
return cal;
}
4、模式类图
5、模式优点
单例模式(Singleton),保证一个类仅有一个实例,并提供一个访问它的全局访问点。
通常我们可以让一个全局变量使得一个对象被访问,但它不能防止你实例化多个对象。一个最好的办法就是,让类自身负责保存它的唯一实例。这个类可以保证没有其他实例可以被创建,并且它可以提供一个访问该实例的方法。
因为Singleton类封装它的唯一实例,这样它可以严格地控制客户怎样访问它以及何时访问它。简单的说就是对唯一实例的受控访问。
单例模式编写最关键的一点是保证方法私有,同时获取实例时保证只有一个对象实例。
6、与类似模式比较
单例模式中主要需要解决的是两个问题:全局访问和实例化控制问题。
静态初始化方式是在自己被加载时就将自己实例化,被形象的称为饿汉式单例类。这种方式是类一加载就实例化的对象,要提前占用系统资源。
原先的单例模式处理方式第要在第一次被引用时,才会将自己实例化,所以称为懒汉式单例类。该方式面临着多线程访问的安全性问题,需要双重锁定这样的处理才能达到保证安全。
谈到单例,我觉得他和静态类还是要说说的,比如数学类这个静态类,静态类一加载,所有方法都会被加载到内存中,他是一种无状态的类,也就是没有实例拥有自己独有的属性,这种类一般就是工具类;然而,单例则不同,他自己内部保存了一份实例,这个实例就是维持了自身的状态,所以这是不同的;另外,因为静态所有的方法都为静态,所以,不能继承,但是单例不一样,单例的方法同样都不是静态的,他是可以继承的。二者共同点是构造方法都私有化。
7、代码实现
1)普通单例(一般单例只要保证构造方法私有),这种单例不安全,原因是多线程访问可能导致产生不止一个实例
/**
* Singleton model (懒汉式单例)
*@author:Heweipo
*@version 1.00
*
*/
public class Singleton {
private static Object lock = new Object();
private static Singleton instance;
private Singleton(){
// do nothing
}
public static Singleton getInstance(){
if(instance == null){
instance = new Singleton();
}
return instance;
}
}
2)多线程单例,Double-Check Locking 双重锁定(在第一次被引用时,才会将自己实例化,所以也称之为 懒汉单例)
/**
* Singleton model (懒汉式单例)
*@author:Heweipo
*@version 1.00
*
*/
public class Singleton {
private static Object lock = new Object();
private static Singleton instance;
private Singleton(){
// do nothing
}
public static Singleton getInstance(){
// 其实可以在方法最外层枷锁,但是那样的话势必会影响性能,因为为空的情况只有一次出现
if(instance == null){
// 枷锁lock,在这之前是可能有多个方法还未初始化单例时进入的,所以后面还有判断一次
synchronized(lock){
// 如果引用为空,那么实例化单例,这里为什么还要判断呢?目的是防止多个方法同时进入
if(instance == null){
//单例首次实例化
instance = new Singleton();
}
}
}
return instance;
}
}
3)静态初始化(在自己被加载时就将自己实例化,被称为饿汉式单例类,提前占用系统资源,不过更加建议使用这种,简洁)
/**
* Singleton model (饿汗式单例)
*@author:Heweipo
*@version 1.00
*
*/
public class Singleton {
// 类加载时就实例化对象
private static Singleton instance = new Singleton();
private Singleton(){
// do nothing
}
public static Singleton getInstance(){
// 直接返回单例
return instance;
}
}