单例的定义与特点
单例(Singleton)模式的定义:指一个类只有一个实例,且该类能自动创建这个实例的一种模式。
在计算机系统中,Windows的回收站,操作系统中的文件系统、多线程中的线程池、显卡的驱动程序对象、打印机的后台处理服务、应用程序的日志对象、数据库的连接池、网站的计数器、Web应用的配置对象,应用程序中的对话框、系统中的缓存等都属于单例场景
应用场景
对于Java来说,单例模式可以保证在一个JVM只存在一个单一实例,单例模式的应用场景主要有以下几个方面。
- 需要频繁创建的一些类,使用单例可以降低系统的内存压力,减少GC
- 某类只要求生成一个对象的时候,如一个班中的班长,每个人的身份证号。
- 某此类创建实例时占用资源比较多。或者实例化耗时比较长,且经常使用。
- 某类需要频繁实例化,而创建对象又频繁被销毁的时候,如多线程的线程池,网络连接池等。
- 频繁访问数据库或文件的对象
- 对于一些硬件级别的操作,或者从系统上讲应当是单一控制逻辑的操作,如果有多个实例,则系统会完全乱套。
- 当对象需要被共享的场景。由于单例模式只允许创建一个对象,共享该对象可以节省内存,并加快对象访问速度。如Web中的配置对象、数据库的连接池等。
常见的两种线程安全常用创建方法
- 饿式模式
最简单的就是饿汉模式,是通过私有静态就是来实现的。在类创建时就已经存在,所以称之为饿汉模。但是注意,不建议占用过多内存资源的单例对象采用饿汉模式,建议采用懒汉模式
public class Singleton1 {
private static Singleton1 instance = new Singleton1();
public static Singleton1 getInstance() {
return instance;
}
}
- 懒汉模式
懒汉模式就在指在需要时才创建,所以称之为懒汉模式。双重校验锁和静态内部类都属于这种模式,本例使用的是内部静态类
public class Singleton2 {
//只有静态类才能拥有静态成员,普通内部类只能定义普通成员
private static class LazyHolder {
private static final Singleton2 INSTANCE = new Singleton2();
}
public Singleton2 getInstance() {
return LazyHolder.INSTANCE;
}
}
- 双重校验锁
再写一个双重校验锁的写法,也是线程安全的,再描述下双重校验的目的
public class Singleton3 {
//通过volatile 保证可见性和有序性
private volatile static Singleton3 instance;
private Singleton3() {
}
public Singleton3 getInstance() {
if (instance == null) {//先校验,减少锁竞争,提升性能
synchronized (Singleton3.class) {
if (instance == null) {//加锁和判断instance不是原子性操作,避免多线程场景的时间片竞争带来的单例失效风险
instance = new Singleton3();
}
}
}
return instance;
}
}