作者简介:大家好,我是smart哥,前中兴通讯、美团架构师,现某互联网公司CTO
联系qq:184480602,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬
学习必须往深处挖,挖的越深,基础越扎实!
阶段1、深入多线程
阶段2、深入多线程设计模式
阶段3、深入juc源码解析
阶段4、深入jdk其余源码解析
阶段5、深入jvm源码解析
码哥源码部分
码哥讲源码-原理源码篇【2024年最新大厂关于线程池使用的场景题】
码哥讲源码【炸雷啦!炸雷啦!黄光头他终于跑路啦!】
码哥讲源码-【jvm课程前置知识及c/c++调试环境搭建】
码哥讲源码-原理源码篇【揭秘join方法的唤醒本质上决定于jvm的底层析构函数】
码哥源码-原理源码篇【Doug Lea为什么要将成员变量赋值给局部变量后再操作?】
码哥讲源码【你水不是你的错,但是你胡说八道就是你不对了!】
码哥讲源码【谁再说Spring不支持多线程事务,你给我抽他!】
终结B站没人能讲清楚红黑树的历史,不服等你来踢馆!
打脸系列【020-3小时讲解MESI协议和volatile之间的关系,那些将x86下的验证结果当作最终结果的水货们请闭嘴】
确保任何情况下都绝对只有一个补全。像这样的被称为单例模式。
public class Hungry {
private static Instance instance = new Instance();
private Hungry() {}
public static Instance getInstance() {
return instance;
}
}
public class Lazy {
private static Instance instance;
private Lazy() {}
public static Instance getInstance() {
if (instance == null) {
instance = new Instance();
}
return instance;
}
}
public class DCL {
private static Instance instance;
private DCL() {}
public static Instance getInstance() {
if (instance == null) {
synchronized (DCL.class) {
if (instance == null) {
instance = new Instance();
}
}
}
return instance;
}
}
注意: 这种方式还是存在线程安全。因为指令重排序,可能会导致
Instance
对象被new出来,并且赋值给instatnce
之后,还没有来得及初始化(执行构造函数中的代码逻辑),就被另一个线程使用了。
Time | ThreadA | ThreadB |
---|---|---|
T1 | 检查到instance为空 | |
T2 | 获取锁 | |
T3 | 再次检查到instance为空 | |
T4 | 为instance分配内存空间 | |
T5 | 将instance指向内存空间 | |
T6 | 检查到instance不为空 | |
T7 | 访问instance(此时对象还未完成初始化) | |
T8 | 初始化instance |
public class DCLPlus {
private volatile static Instance instance;
private DCLPlus() {}
public static Instance getInstance() {
if (instance == null) {
synchronized (DCLPlus.class) {
if (instance == null) {
instance = new Instance();
}
}
}
return instance;
}
}
public enum EnumSingleton {
INSTANCE;
public Instance getInstance() {
return new Instance();
}
}
public class StaticInner {
private StaticInner() {}
private static class SingleonHolder {
private static final Instance instance = new Instance();
}
public static Instance getInstance() {
return SingleonHolder.instance;
}
}
SingleonHolder
是一个静态内部类,当外部类StaticInner
被加载的时候,并不会创建该实例对象。只有当调用getInstance()
方法时,SingleonHolder
才会加载,这个时候才会创建instance
。insance 的唯一性、创建过程的线程安全性,都由JVM 来保证。所以,这种实现方法既保证了线程安全,又能做到延迟加载。