单例模式其实就是保证一个类的实例只有一个,并提供一个全局访问点,常用于管理共享资源,例如线程池,数据库连接池等等。
单例模式最重要的特点就是,
1.构造方法是private的(无法从其他类实例化该类的对象)
2.类内部有一个该类的静态的实例以及一个产生该类实例的静态方法。(提供一个全局访问点,由这个类来实例化,在要用到这个实例的时候才实例化,也是所谓的延迟实例化)
public class Singleton
{
public static Singleton uniqueInstance;
private Singleton()
{
}
public static Singleton getInstance()
{
if(uniqueInstance==null)
{
uniqueInstance=new Singleton();
}
return uniqueInstance;
}
}
但这样的单例模式有一个问题,就是当在多线程中运行时,会出现产生两个对象的情况。
解决办法是:
1.将getInstance()方法改成同步方法,即在该方法前面加上synchronized关键字
这个方法避免了多个线程同时访问该方法,但是,实际上,只有在这个对象没被实例化之前才需要同步,而一旦使用同步方法,则会大大减低程序的效率
public static synchronized Singleton getInstance()
{
if(uniqueInstance==null)
{
uniqueInstance=new Singleton();
}
return uniqueInstance;
}
2.“急切”创建实例,而不是“延迟实例化”
在静态初始化块中实例化对象,使得jvm在一开始的时候就实例化这个对象。
例如:
public class SingletonEagerInit
{
public static Singleton uniqueInstance=new Singleton();
private Singleton()
{
}
public static Singleton getInstance()
{
return uniqueInstance;
}
}
3.利用双重检查加锁,首先检查实例是否创建了,如果没有创建,才进行同步
public static Singleton getInstance()
{
if(uniqueInstance==null)
{
synchronized(SingletonDoubleCheck.class)
{
if(uniqueInstance==null)
{
uniqueInstance=new Singleton();
}
}
}
return uniqueInstance;
}
单例模式虽然看起来很简单,但要很优雅地使用还是需要好好研究,特别是涉及到多线程的问题上。
原文:http://www.cnblogs.com/qingfei1994/p/4243488.html