学习设计模式与范式:创建型中的单例模式
1.单例的定义
单例设计模式(Singleton Design Pattern)理解起来非常简单。一个类只允许创建一个对象
(或者叫实例),那这个类就是一 个单例类 ,这种设计模式就叫作单例设计模式,简称单例模式。
2.单例的用处
从业务概念上,有些数据在系统中只应该保存一份, 就比较适合设计为单例类。比如,統的
配置信息类。除此之外,我们还可以使用单例解决资源访问冲突的问题。
3.单例的实现
单例有下面几种经典的实现方式。
●饿汉式
饿汉式的实现方式,在类加载的期间,就已经将instance静态实例初始化好了,所以,
instance实例的创建是线程安全的。过,这样的实现方式不支持延迟加载实例。
public class IdGenerator {
private AtomicLong id = new AtomicLong(0);
private static final IdGenerator instance = new IdGenerator();
private IdGenerator() {}
public static IdGenerator getInstance() {
return instance;
}
public long getId() {
return id.incrementAndGet();
}
}
●懒汉式
懒汉式相对于饿汉式的优势是支持延迟加载。这种实现方式会导致频繁加锁、释放锁,以及并发度低等问题,频繁的调用会产生性能瓶颈。
public class IdGenerator {
private AtomicLong id = new AtomicLong(0);
private static IdGenerator instance;
private IdGenerator() {}
public static synchronized IdGenerator getInstance() {
if (instance == null) {
instance = new IdGenerator();
}
return instance;
}
public long getId() {
return id.incrementAndGet();
}
}
●双重检测
双重检测实现方式既支持延迟加载、又支持高并发的单例实现方式。只要instance被创建之后,再调用getInstance()函数都不会进入到加锁逻辑中。所以,这种实现方式解决了懒汉式并发度低的问题。
public class IdGenerator {
private AtomicLong id = new AtomicLong(0);
private static IdGenerator instance;
private IdGenerator() {}
public static IdGenerator getInstance() {
if (instance == null) {
synchronized(IdGenerator.class) { // 此处为类级别的锁
if (instance == null) {
instance = new IdGenerator();
}
}
}
return instance;
}
public long getId() {
return id.incrementAndGet();
}
}
●静态内部类
利用Java的静态内部类来实现单例。这种实现方式,既支持延迟加载,也支持高并发,实现
起来也比双重检测简单。
public class IdGenerator {
private AtomicLong id = new AtomicLong(0);
private IdGenerator() {}
private static class SingletonHolder{
private static final IdGenerator instance = new IdGenerator();
}
public static IdGenerator getInstance() {
return SingletonHolder.instance;
}
public long getId() {
return id.incrementAndGet();
}
}
●枚举
最简单的实现方式,于枚举类型的单例实现。这种实现方式通过Java枚举类型本身的特
性,保证了实例创建的线程安全性和实例的唯一性。
public enum IdGenerator {
INSTANCE;
private AtomicLong id = new AtomicLong(0);
public long getId() {
return id.incrementAndGet();
}
}