文章目录
- 一、概述
- 二、UML图
- 三、适用场景
- 四、特点
- 五、单例模式的7种实现形式
- 六、spring中的单例
- 七、单例模式的破坏
- 参考
- 【1】[破坏单例模式](https://mp.weixin.qq.com/s?__biz=MzI3NzE0NjcwMg==&mid=2650128883&idx=1&sn=6f284f0f3d00bf36ad8933907d45c630&chksm=f36bdcd2c41c55c4d7c0c67caf96cc8bce0ec4ae3598b2a1516c0c92d6573a99bd4850f4d43b&scene=27#wechat_redirect)
- 【2】[为什么推荐使用枚举来创建单例](https://mp.weixin.qq.com/s?__biz=MzI3NzE0NjcwMg==&mid=2650121482&idx=1&sn=e5b86797244d8879bbe9a69fb72641b5&chksm=f36bb82bc41c313d739f485383d3a868a79020c995ee86daef026a589f4782916c42a8d3f6c7&scene=27#wechat_redirect)
一、概述
单例模式是指在整个软件系统中,一个类只有一个实例对象,并且该类需要提供访问该实例对象的方法(静态方法)
二、UML图
三、适用场景
- 当需要频繁的创建和销毁对象,并且创建对象耗时较长,资源消耗较多,但是会经常使用到的对象,就需要使用单例模式
四、特点
- 一个类只有一个实例
五、单例模式的7种实现形式
1)饿汉式(静态常量)
- 示例
public class Singleton_01 {
private static final Singleton_01 instance = new Singleton_01();
private Singleton_01(){}
public static Singleton_01 getInstance() {
return instance;
}
}
class Test_01 {
public static void main(String[] args){
Singleton_01 instance1 = Singleton_01.getInstance();
Singleton_01 instance2 = Singleton_01.getInstance();
System.out.println(instance1 == instance2);
System.out.println("instance1 hash code: " + instance1.hashCode());
System.out.println("instance1 hash code: " + instance2.hashCode());
}
}
- 优缺点说明
- 优点:写法简单,在类加载完成后就完成了初始化,避免了线程同步的问题
- 缺点:在类加载完成后完成了实例化,没有达到Lazy Loading的效果,如果系统从始至终没有使用过该实例,便造成了内存的浪费
- 说明:这种方式基于ClassLoader机制(ClassLoader机制本身是线程安全的)避免了多线程的同步问题,但是,其实例会在类加载完成之后立即完成了初始化;如果由于其他原因,导致了该类的加载,便会导致该类的实例被初始化,没有达到Lazy Loading的效果
- 结论:写法简单,但是可能会造成内存浪费;如果确定实例会使用,建议使用
2)饿汉式(静态代码块)
- 示例
public class Singleton_02 {
private static Singleton_02 instance;
static {
instance = new Singleton_02();
}
private Singleton_02(){}
public static Singleton_02 getInstance() {
return instance;
}
}
class Test_02 {
public static void main(String[] args){
Singleton_02 instance1 = Singleton_02.getInstance();
Singleton_02 instance2 = Singleton_02.getInstance();
System.out.println(instance1 == instance2);
System.out.println("instance1 hash code: " + instance1.hashCode());
System.out.println("instance1 hash code: " + instance2.hashCode());
}
}
- 优缺点说明
- 同饿汉式静态常量的方式一样,只不过把示例的初始化过程放在了静态代码块中;
3)懒汉式(线程不安全)
- 示例
public class Singleton {
private static Singleton instance;
private Singleton(){}
public static Singleton getInstance() {
if (instance == null)
instance = new Singleton();
return instance;
}
}
class Test_02 {
public static void main(String[] args){
// 多线程下测试线程不安全问题(执行多次, 会发现输出的hashCode不同)
for (int i = 0; i < 5; i++) {
new Thread(() -> {
Singleton instance = Singleton.getInstance();
System.out.println(instance.hashCode());
}, "T-" + i).start();
}
}
}
- 优缺点说明
- 起到了Lazy Loading的效果,但是线程不安全
- 在多线程环境下,两个线程同时执行到
if (instance == null)
,若为true,则会创建两个不同的实例- 多线程环境下不能使用该形式去创建单例
4)懒汉式(线程安全,同步方法)
- 示例
public class Singleton {
private static Singleton instance;
private Singleton(){}
public synchronized static Singleton getInstance() {
if (instance == null)
instance = new Singleton();
return instance;
}
}
class Test {
public static void main(String[] args){
// 多线程下测试线程不安全问题(执行多次, 线程安全)
for (int i = 0; i < 5; i++) {
new Thread(() -> {
Singleton instance = Singleton.getInstance();
System.out.println(instance.hashCode());
}, "T-" + i).start();
}
}
}
- 优缺点说明
- 使用synchronized修饰静态方法, 保证了线程安全
- 但是每次调用getInstance方法时, 都需要获取锁, 增加等待时间, 效率较低
5)双重检查
public class Singleton {
private volatile static Singleton instance;
private Singleton(){}
public static Singleton getInstance() {
if (instance == null) {
synchronized (Singleton.class) {
if (instance == null){
instance = new Singleton();
}
}
}
return instance;
}
}
class Test {
public static void main(String[] args){
// 多线程下测试线程不安全问题(执行多次, 线程安全)
for (int i = 0; i < 5; i++) {
new Thread(() -> {
Singleton instance = Singleton.getInstance();
System.out.println(instance.hashCode());
}, "T-" + i).start();
}
}
}
- 优缺点说明
- Double-Check概念是多线程开发中常用的方式;示例中进行了两次if(instance == null)的检查,第一次检查是为了在创建对象时,判断对象是否为空,若为空则进行创建对象;第二次检查,并且加了同步锁,为了避免多线程情况下,两个线程同时通过了第一次检查,都进入了创建对象的步骤,第二次加锁检查,就只能允许一个线程进行对象的创建
- 确保了实例化代码只需要执行一次,后面再次访问时,直接返回实例化之后的对象
- 线程安全,延迟加载,效率较高;实际开发中推荐使用
6)静态内部类
public class Singleton {
private volatile static Singleton instance;
private Singleton(){}
public static Singleton getInstance() {
return InnerSingleton.INSTANCE;
}
private static class InnerSingleton {
private static final Singleton INSTANCE = new Singleton();
}
}
class Test {
public static void main(String[] args){
// 多线程下测试线程不安全问题(执行多次, 线程安全)
for (int i = 0; i < 5; i++) {
new Thread(() -> {
Singleton instance = Singleton.getInstance();
System.out.println(instance.hashCode());
}, "T-" + i).start();
}
}
}
- 有缺点说明
- 采用了类装载的机制保证了线程安全
- 静态内部类InnerSingleton在Singleton类被加载时不会立即被初始化,而是在调用getInstance方法时才会对InnerSingleton进行加载操作,加载InnerSingleton会对静态常量进行初始化操作
- 利用静态内部类延迟加载的特性实现了懒加载,效率较高
7)枚举
- 示例
public class Test {
public static void main(String[] args){
Singleton instance = Singleton.INSTANCE;
instance.method();
}
}
enum Singleton {
INSTANCE;
public void method() {
System.out.println("Do Something!");
}
}
- 优缺点说明
- 枚举保证了线程安全
- 枚举可以避免反序列化时重复创建对象
六、spring中的单例
1、spring中单例bean的创建流程
1.1 注册初始的BeanDefinition,此时scope属性为空
- 调用DefaultListableBeanFactory#registerBeanDefinition方法
- 此时只是初始化了部分BeanDefinition(Spring的基础Bean,及main方法所在的Bean)
1.2 初始化scope
- 调用AbstractBeanFactory#getMergedBeanDefinition,将scope为空的BeanDefinition设置为singleton
if (!StringUtils.hasLength(mbd.getScope())) {
mbd.setScope(SCOPE_SINGLETON);
}
1.3 注册业务Bean·
七、单例模式的破坏
1、反射的方式破坏单例
☞ 原理
通过反射的方式,获取到私有的构造器,设置可以访问私有构造器的权限,然后通过反射调用的方式去创建实例
public class Reflect {
public static void main(String[] args) throws Exception {
Class<?> clazz = Class.forName("com.lizza.d1_singleton.singleton_1.Singleton");
Constructor<?> constructor = clazz.getDeclaredConstructor();
constructor.setAccessible(true);
Singleton instance_1 = (Singleton) constructor.newInstance();
Singleton instance_2 = (Singleton) constructor.newInstance();
System.out.println(instance_1.hashCode());
System.out.println(instance_2.hashCode());
}
}
☞ 如何避免
在私有构造器中进行判空处理
private Singleton(){
if (instance != null) {
throw new RuntimeException("instance is must be null!");
}
}
2、序列化和反序列化破坏单例
☞ 原理
通过对象序列化可以得到不同的克隆对象,以此来破坏单例
public class Serialize {
public static void main(String[] args){
// 传统单例
com.lizza.d1_singleton.singleton_1.Singleton instance = com.lizza.d1_singleton.singleton_1.Singleton.getInstance();
// 枚举: 可以避免序列化破坏单例的情况
// com.lizza.d1_singleton.singleton_7.Singleton instance = com.lizza.d1_singleton.singleton_7.Singleton.INSTANCE;
Object object = clone(instance);
System.out.println(instance.getClass() + ", " + instance.hashCode());
System.out.println(instance.getClass() + ", " + object.hashCode());
}
public static Object clone(Object object) {
ObjectInputStream is = null;
ObjectOutputStream os = null;
try {
ByteArrayOutputStream bos = new ByteArrayOutputStream();
os = new ObjectOutputStream(bos);
os.writeObject(object);
is = new ObjectInputStream(new ByteArrayInputStream(bos.toByteArray()));
return is.readObject();
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (os != null) {
os.close();
}
if (is != null) {
is.close();
}
} catch (Exception e) {
e.printStackTrace();
}
}
return null;
}
}
☞ 解决方式
在Singleton中增加readResolve方法,反序列化时会判断是否有readResolve方法, 若有则通过该方法创建对象
/** 反序列化时会判断是否有readResolve方法, 若有则通过该方法创建对象 **/
private Object readResolve() {
System.out.println(">>> readResolve is run!");
return getInstance();
}