首先给出代码:
- //基于类初始化的线程安全的单例
- class SingleTon4{
-
- private SingleTon4(){}
-
- private static class InnerClass{
- private static SingleTon4 instance= new SingleTon4();
- }
- public static SingleTon4 getInstance(){//如果没有到这里,那么不会加载上面的内部类
- return InnerClass.instance; //这里将导致InstanceHolder类被初始化
- }
- }
静态内部类的优点是:外部类加载时并不需要立即加载内部类,内部类不被加载则不去初始化INSTANCE,故而不占内存。具体来说当SingleTon第一次被加载时,并不需要去加载SingleTonHoler,只有当getInstance()方法第一次被调用时,使用INSTANCE的时候,才会导致虚拟机加载SingleTonHoler类。这种方法不仅能确保线程安全,也能保证单例的唯一性,同时也延迟了单例的实例化。
那么他是如何实现线程安全的?
首先要了解类加载过程中的最后一个阶段:即类的初始化,类的初始化阶本质就是执行类构造器的
那么回到这个代码中,这里的静态变量的赋值操作进行编译之后实际上就是一个
那么再增加一句,之所以这里变量定义的时候不需要volatile,因为只有一个线程会执行具体的类的初始化代码
此外,如果讲得再细致一点,可以参考下面的第二个网站,简单说一下:
比如有一个类T,那么什么时候会进行类T的初始化?有以下5种情况:
而我们前面的代码则对应了第四种情况,一个静态字段被使用,因此此时则会进行静态内部类的初始化。又因为java是多线程语言,作者也考虑到了可能存在“多个线程在同一时间尝试去初始化同一个类”的情况,因此java规定,对于每一个类或接口 C,都有一个唯一的初始化锁 LC 与之对应。
JVM 在类初始化期间会获取这个初始化锁,并且每个线程至少获取一次锁来确保这个类已经被初始化过了。(但是只有第一个获得锁的线程,会执行初始化,执行完之后会设置一个标志位,表示已经初始化完成,后面其他的线程再次获得锁,检查标志位,发现已经初始化完了,直接释放锁,不会再次执行初始化。)
参考:https://blog.csdn.net/qq_35590091/article/details/107348114
《深入理解JAVA虚拟机》