什么是单例模式?(详解)
单例模式,顾名思义就是程序在运行的过程中,有且只有一个实例。
它必须满足三个关键点:
1).一个类只有一个实例。这是满足单例模式最基本的要求,若要满足这个关键点,只能提供私有的构造器,即保证不能随意创建该类的实例。
2). 它必须自行创建这个实例。对于这一点,正是体现了:单例模式的“有且仅有一个实例”的这一特性。我们要保证唯一性,也就意味着必须要提供一个实例,那么就需要它自行创建,定义一个ConfigManager类型的静态的私有对象,以便向外界提供该类实例时使用。
3).它必须自行向整个系统提供这个实例。最后一点也是至关重要的,外界需要获取并使用这个单例类的实例,但是由于该类的构造器是私有的,外界无法通过new去获取它的实例,那么就必须提供一个静态的公有方法,该方法创建或者获取它本身的静态私有对象并返回。
在java语言中,单例分为懒汉式,饿汉式。
懒汉式:指全局的单例实例在第一次使用是创建。
// 懒汉式
public class Singleton {
// 私有构造
private Singleton() {}
private static Singleton single = null;
public static Singleton getInstance() {
//判断是否存在,不存在就创建
if(single == null){
single = new Singleton();
}
return single;
}
}
-
懒汉式的线程是不安全的,
-
线程不安全原因:我们假设有多个线程1,线程2都需要使用这个单例对象。而恰巧,线程1在判断完single ==null后突然交换了cpu的使用权,变为线程2执行,由于single 仍然为null,那么线程2中就会创建这个Singleton的单例对象。之后线程1拿回cpu的使用权,而正好线程1之前暂停的位置就是判断single 是否为null之后,创建对象之前。这样线程1又会创建一个新的Singleton对象。
-
解决办法:加同步锁 synchronized
加同步锁之后的代码
public class SingleDemo {
private static SingleDemo s = null;
private SingleDemo(){}
public static synchronized SingleDemo getInstance(){
if(s == null){
s = new SingleDemo();
}
return s;
}
}
饿汉式:全局的实例在类加载的时候构建
// 饿汉式
public class Singleton1 {
// 私有构造
private Singleton1() {}
private static Singleton1 single = new Singleton1();
// 静态工厂方法
public static Singleton1 getInstance() {
return single;
}
}
- 饿汉式单例在类加载初始化时就创建好一个静态的对象供外部使用,除非系统重启,这个对象不会改变,所以本身就是线程安全的。
- Singleton通过将构造方法限定为private避免了类在外部被实例化,在同一个虚拟机范围内,Singleton的唯一实例只能通过getInstance()方法访问。