静态内部类懒加载实现原理

本文探讨了静态内部类如何实现单例的懒加载模式,结合Java虚拟机的类初始化规则,解释了为何这种方式在多线程环境中能确保线程安全并实现延迟初始化。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言

最近在看《深入理解Java虚拟机》一书,看到了类初始化的五种情况。联想到之前看到过通过静态内部类来实现单例懒加载的模式,当时不理解其原理,现在特此记录一下。

由传统单例引起

传统的单例模式分为饿汉,懒汉模式。因为synchronized 关键字,导致性能被浪费。在实例化之后的对象读取,希望是无锁的。所以引入“双检锁模式”

public class Singleton{
    private static Singleton singleton;

    private Singleton() {
    }

    public synchronized static Single newInstance() {
        if (singleton== null) {
            singleton= new Singleton();
        }
        return singleton;
    }
}

双检锁

但是,这种双检锁模式也会存在问题,多线程情况下,会出现某个线程获取到的实例对象虽然被赋予默认值,但其初始化还未完成的情况。这种问题的解决办法则是添加volatile关键字,但volatile也会有性能损耗。

public class DoubleCheckedLock {

    private static DoubleCheckedLock instance;

    public static DoubleCheckedLock getInstance(){
        if (null == instance){
            synchronized (DoubleCheckedLock.class){
                if (null == instance){
                    instance = new DoubleCheckedLock();
                }
            }
        }
        return instance;
    }

}

静态内部类

这种方式通过jvm来保证其线程安全,而懒加载的机制则是依靠虚拟机规范制度的类“初始化”规则保证

public class Singleton {
    private static class SingletonHolder {
        private static Singleton singleton = new Singleton();
    }

    private Singleton() {
    }

    public static Singleton newInstance() {
        return SingletonHolder.singleton;
    }
}

类初始化规则

虚拟机规范则严格规定了有且只有5种情况必须立即对类进行“初始化”(而加载、验证、准备自然需要
在此之前开始):

  • 遇到new、getstatic、putstatic或invokestatic这4条字节码指令时,如果类没有进行过初
    始化,则需要先触发其初始化。生成这4条指令的最常见的Java代码场景是:使用new关键字
    实例化对象的时候、读取或设置一个类的静态字段(被final修饰、已在编译期把结果放入常
    量池的静态字段除外)的时候,以及调用一个类的静态方法的时候。
  • 使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,
    则需要先触发其初始化。
  • 当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父
    类的初始化。
  • 当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个
    类),虚拟机会先初始化这个主类。
  • 当使用JDK 1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后
    的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄
    所对应的类没有进行过初始化,则需要先触发其初始化。

参考:《深入理解Java虚拟机》

### Java静态内部类的加载机制 Java中的静态内部类是一种特殊的嵌套类,它通过`static`关键字声明。这种设计不仅提高了代码的组织能力,还增强了程序的功能模块化程度。当涉及到静态内部类的加载机制时,主要依赖于Java虚拟机(JVM)的行为及其对类加载过程的理解。 #### 类加载时机 根据JVM规范,只有在首次主动使用某个类时才会触发其初始化[^4]。对于外部类和静态内部类的关系而言: - 外部类的加载并不会自动引发静态内部类的加载。 - 只有当程序显式访问静态内部类的成员或创建其实例时,才触发静态内部类的加载与初始化。 例如: ```java class Outer { static class Inner { static int value = 42; } } public class Test { public static void main(String[] args) { System.out.println(Outer.Inner.value); // 此处会触发Inner类的加载 } } ``` 上述代码中,仅当`System.out.println(Outer.Inner.value)`执行时,`Outer.Inner`会被加载并完成初始化操作。 --- #### 静态内部类的加载流程 静态内部类作为独立的顶层类存在,尽管逻辑上属于外部类的一部分,但在实际编译过程中,它们被单独编译为`.class`文件。因此,静态内部类的加载遵循标准的类加载机制,具体分为以下几个阶段: 1. **加载** JVM会在第一次需要使用该类时将其字节码加载到内存中。此阶段涉及定位目标类的二进制数据源(通常是磁盘上的`.class`文件),并通过类加载器将这些数据转换为方法区内的运行时表示形式[^3]。 2. **验证** 对已加载的类元数据进行校验,确保其结构符合JVM规范的要求,不会危及虚拟机的安全性。 3. **准备** 创建类或接口所需的静态字段,并为其分配初始值(通常为零值)。此时尚未执行任何用户定义的初始化代码[^5]。 4. **解析** 将符号引用替换为直接引用的过程。在此期间,如果静态内部类中有对外界其他类型的引用,则需进一步确认那些类型是否已被正确加载。 5. **初始化** 执行类构造器 `<clinit>` 方法的内容,其中包括所有静态变量赋值语句以及静态代码块中的指令。值得注意的是,由于静态内部类本身不持有对外围实例的隐含引用,所以它可以安全地参与单例模式的设计[^2]。 --- #### 单例模式下的静态内部类优势 利用静态内部类实现单例模式具有延迟加载的特点。这是因为静态内部类只有在其成员被初次访问时才会被加载,从而避免了不必要的开销。下面是一个典型的例子: ```java public class Singleton { private Singleton() {} private static class Holder { private static final Singleton INSTANCE = new Singleton(); } public static Singleton getInstance() { return Holder.INSTANCE; // 触发Holder类的加载 } } ``` 在这个案例里,直到调用 `Singleton.getInstance()` 方法之前,`Holder` 类都不会被加载,进而实现了懒汉式的单例效果。 --- #### 自定义类加载器的影响 假如应用程序引入了自定义类加载器来处理某些特定需求,则需要注意不同类加载器之间可能产生的冲突问题。即使两个类拥有相同的全限定名,只要由不同的类加载器加载,就会被视为完全不同的类型。这种情况可能导致诸如 `ClassNotFoundException` 或者无法成功转型等问题发生。 --- ### 总结 综上所述,Java静态内部类的加载机制本质上与其他普通类无异,只是因其特殊地位使得开发者能够更加灵活地控制何时对其进行初始化。理解这一原理有助于优化性能敏感的应用场景,同时也提醒我们在复杂环境中谨慎对待多层嵌套关系带来的潜在风险。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值