From 1872079592273a44502923a10e4243c8c8f0efe8 Mon Sep 17 00:00:00 2001 From: Zoctan <752481828@qq.com> Date: Sat, 22 Sep 2018 12:57:45 +0800 Subject: [PATCH] =?UTF-8?q?Update=20=E8=AE=BE=E8=AE=A1=E6=A8=A1=E5=BC=8F.m?= =?UTF-8?q?d?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- notes/设计模式.md | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/notes/设计模式.md b/notes/设计模式.md index c177ff92..ec581313 100644 --- a/notes/设计模式.md +++ b/notes/设计模式.md @@ -145,15 +145,15 @@ if (uniqueInstance == null) { uniqueInstance 采用 volatile 关键字修饰也是很有必要的。`uniqueInstance = new Singleton();` 这段代码其实是分为三步执行。 -1. 为 uniqueInstance 分配内存空间 +1. 为 uniqueInstance 分配栈内存空间 2. 初始化 uniqueInstance -3. 将 uniqueInstance 指向分配的内存地址 +3. 将 uniqueInstance 指向对象所分配的堆内存地址 但是由于 JVM 具有指令重排的特性,执行顺序有可能变成 1>3>2。指令重排在单线程环境下不会出先问题,但是在多线程环境下会导致一个线程获得还没有初始化的实例。例如,线程 T1 执行了 1 和 3,此时 T2 调用 getUniqueInstance() 后发现 uniqueInstance 不为空,因此返回 uniqueInstance,但此时 uniqueInstance 还未被初始化。 使用 volatile 可以禁止 JVM 的指令重排,保证在多线程环境下也能正常运行。 -#### Ⅴ 静态内部类实现 +#### 静态内部类实现 当 Singleton 类加载时,静态内部类 SingletonHolder 没有被加载进内存。只有当调用 `getUniqueInstance()` 方法从而触发 `SingletonHolder.INSTANCE` 时 SingletonHolder 才会被加载,此时初始化 INSTANCE 实例,并且 JVM 能确保 INSTANCE 只被实例化一次。