• JVM -- 类的初始化


    《深入理解Java虚拟机》 第二版中介绍到了类的加载过程。

    一个类从加载入内存到卸载出内存为止,整个生命周期包括:

    Loading(加载)-----Verification(验证)-----Preparation(准备)-----Resolution(解析)-----Initialization(初始化)-----Using(使用)----Unloading(卸载)

    在Initalization 阶段,虚拟机规范严格定一个5种必须立即对类初始化的情况。

    1、遇到new ,getstatic putstatic invokestatic

    2、使用java.lang.reflect 包中的方法对类进行反射调用的时候,如果没有初始化,要首先触发其初始化。

    3、初始化一个类的时候,如果发现其父类还未初始化,那么首先初始化其父类。

    4、当虚拟机启动的时候,用户需要指定一个需要执行的主类,然后虚拟机才会初始化这个主类。

    5、JDK1.7的动态语言支持,如果一个java.lang.invoke.MethodHandle 实例最后解析结果REF_getStatic,REF_putStatic,REF_invokeStatic 的方法句柄,并且这个方法句柄对应的类没有进行初始化,则需要将其初始化。

    除此之外,所有引用类的方式都不会触发初始化。---被动引用。

    1、通过子类调用父类中定义的静态字段,不会导致子类的初始化

     1 package volshell;
     2 
     3 public class SuperCalss {
     4     static {
     5         System.out.println("super class init");
     6     }
     7     public static int value = 23;
     8 }
     9 
    10 package volshell;
    11 
    12 public class SubClass extends SuperCalss {
    13     static {
    14         System.out.println("subclass init");
    15     }
    16 }
    17 
    18 package volshell;
    19 
    20 public class Demo1 {
    21     public static void main(String[] args) {
    22         System.out.println(SubClass.value);
    23     }
    24 }

    输出:

      super class init
      23
    对于静态字段,只有直接定义该字段的类才会被初始化,因此通过子类的来引用父类定义的静态字段,并不会触发子类的初始化。

    2、使用数组来定义引用类,不会触发该类的初始化。

      还是使用上面的代码在main方法中添加

      SuperCalss[]subClass = new SuperCalss[10];

    执行之后不会有任何的输出。

    3、常量在编译接阶段就会存入调用类的常量池中,本质上没有直接引用定义常量的类,因此不会触发被引用的类的初始化。

      同样还是使用上面的例子 ,将superClass 中的 value 设置为public static final int value=23; 设置为常量

    在主类中调用SuperClass.value

     执行结果;

      23

    在编译阶段,通过常量的传播优化,已将常量value存入Demo1 的常量池中了。

  • 相关阅读:
    java 菜单
    QT 让信号自由飞翔(骚操作)
    QT editLine 无法输入的问题
    易经初学体会
    Cgroup
    springboot pom 引用集合
    使用ab测试工具 进行并发测试
    intellij 设置-试验过的
    【iis错误码】IIS 服务 这些年遇到的错误码
    101个创业失败案例背后的20大原因
  • 原文地址:https://www.cnblogs.com/plxx/p/4528253.html
Copyright © 2020-2023  润新知