定义
确保一个类只有一个实例,且对外公开这个唯一的实例。保证在这个系统中唯一。
- 所以我们需要保证这个类的构造方法不能公开,是类私有的,不能被外界实例化。且这个类只有一个实例,这个实例属于当前类,也就是说这个实例是这个类的类成员变量,也就是静态变量。
作用
单例模式的主要作用是确保一个类只有一个实例存在,可以用作序列号生成,web页面计数器等,spring中的bean也是使用单例模式实现的。同时若是创建一个对象消耗很多资源,如数据库连接,访问io等,我们也可以用单例模式,减少资源消耗。
实现
饿汉式
public class Singleton{
private final static Singleton singleton = new Singleton();
private Singleton(){}
public static Singleton getSingleton(){
return singleton;
}
}
饿汉式线程安全,在类加载的时候就已经创建唯一的单例。
懒汉式
public class Singleton{
private final static Singleton singleton;
private Singleton(){}
public static Singleton getSingleton(){
if(singleton==null){
singleton = new Singleton();
}
return singleton;
}
}
懒汉式线程不安全,它在第一次获取单例的时候才实例化,可能会出现A和B同时获取单例且这个单例还没实例化的情况,会被实例化多次。
解决方法
- 直接对获取方法加锁
public class Singleton{
private final static Singleton singleton;
private Singleton(){}
public static synchronized Singleton getSingleton(){
if(singleton==null){
singleton = new Singleton();
}
return singleton;
}
}
// 效率较慢,性能差
- 双重检查锁
public class Singleton{
private final static volatile Singleton singleton;
private Singleton(){}
public static Singleton getSingleton(){
if(singleton==null){
synchronized(Singleton.class){
if(singleton==null){
singleton = new Singleton();
}
}
}
return singleton;
}
}
/*
双重检查锁要对实例对象singleton加上volatile关键字修饰,防止指令重排。
因为new Singleton分为3步
1.分配内存
2.初始化对象
3.指向刚分配的地址
若不使用volatile关键字,有可能会先执行1,3导致还没初始化对象,导致返回null。
*/
- 静态内部类
public class Singleton{
private static class SingletonHolder{
private final static Singleton singleton = new Singleton();
}
private Singleton(){}
public static Singleton getSingleton(){
return SingletonHolder.singleton;
}
}
还有一种枚举方式,但是个人不是特别理解,故先不写出了。
总结
单例模式是保证一个类的实例系统全局唯一,是一种设计思想。