单件模式可以得到一个唯一的对象,因为它没有公开的构造函数,其他类使用的时候只能通过请求的方式得到,而不能通过构造函数的方式得到。
单件模式确保一个类只有一个实例,并提供一个全局访问点。(延迟实例化)
多线程时,还是可能出现创建多个实例的情况。可以在getInstance前加上synchronize避免这个问题
但是同步会降低性能
public class Singleton {
private static Singleton uniqueInstance;
private Singleton(){}
public static synchronized Singleton getInstance(){
if(uniqueInstance==null){
uniqueInstance=new Singleton();
}
return uniqueInstance;
}
}
或者使用急切创建实例的方式。这个做法,JVM保证任何线程访问此变量前,一定先创建此实例
public class Singleton {
private static Singleton uniqueInstance=new Singleton();
private Singleton(){}
public static synchronized Singleton getInstance(){
return uniqueInstance;
}
}
利用双重检查加锁,首先检查实例是否已经创建了,如果尚未创建,才进行同步,这样一来,只是在第一次创建的时候同步。效率会提高。
public class Singleton {
private volatile static Singleton uniqueInstance=new Singleton();
//使用volatile关键词确保:当uniqueInstance变量被初始化成Singleton实例时。
// 多个线程正确的处理uniqueInstance变量
private Singleton(){}
public static Singleton getInstance(){
if(uniqueInstance==null){//第一次彻底执行。
synchronized (Singleton.class) {//保证多线程的正确性
if(uniqueInstance==null){
uniqueInstance=new Singleton();
}
}
}
return uniqueInstance;
}
}
注意,如果程序中存在多个类加载器,可能会出现多个单件共存的现象。解决方法:自行指定类加载器,并指定同一个类加载器。
单件的继承可能会出现很多问题,会改变构造器的权限,使得实例化的方式不唯一了。不推荐继承。