欢迎访问悦橙教程(wld5.com),关注java教程。悦橙教程  java问答|  每日更新
页面导航 : > > 文章正文

【Java】Java初始化过程总结,

来源: javaer 分享于  点击 45120 次 点评:243

【Java】Java初始化过程总结,


概述

  • Java字节代码:byte[]
  • Java类在JVM的表现形式:Class类的对象;

Java源代码被编译成class字节码 :


Java字节代码 --> Class类的对象:


Java类的加载

作用

把Java字节码转换成JVM中的java.lang.Class类的对象;

类加载器分类

双亲委派模型工作过程:

当一个类加载器收到类加载任务,优先交给其父类加载器去完成,因此最终加载任务都会传递到顶层的启动类加载器,只有当父类加载器无法完成加载任务时,才会尝试执行加载任务。

双亲委派模型有什么好处?

比如位于rt.jar包中的类java.lang.Object,无论哪个加载器加载这个类,最终都是委托给顶层的启动类加载器进行加载,确保了Object类在各种加载器环境中都是同一个类。

重要特征


Java类的链接

Java类的链接指的是将Java类的二进制代码合并到JVM的运行状态之中的过程。

包含的步骤:

private static int var = 100;

准备阶段完成后,var值为0,而不是100。在初始化阶段,才会把100赋值给val,但是有个特殊情况:

private static final int VAL= 100;

在编译阶段会为VAL生成ConstantValue属性,在准备阶段虚拟机会根据ConstantValue属性将VAL赋值为100。

  • 符号引用使用一组符号来描述所引用的目标,可以是任何形式的字面常量,定义在Class文件格式中。
  • 直接引用可以是直接指向目标的指针、相对偏移量或则能间接定位到目标的句柄。

Java类的初始化

初始化阶段是执行类构造器clinit方法的过程,clinit方法由类变量的赋值动作和静态语句块按照在源文件出现的顺序合并而成,该合并操作由编译器完成。

  • 初始化过程的主要操作是执行静态代码块初始化静态域
  • 在一个类被初始化之前,它的直接父类也需要被初始化。

类初始化场景

虚拟机中严格规定了有且只有5种情况必须对类进行初始化。

不会触发类初始化的情况

  • 通过子类引用父类的静态字段,只会触发父类的初始化,而不会触发子类的初始化
class Parent {
    static int a = 100;
    static {
        System.out.println("parent init!");
    }
}
 
class Child extends Parent {
    static {
        System.out.println("child init!");
    }
}
 
public class Init{  
    public static void main(String[] args){  
        System.out.println(Child.a);  //不会初始化类Child
    }  
}

输出结果为:parent init! 不会初始化Child类。

  • 定义对象数组,不会触发该类的初始化。
 public class Init{  
    public static void main(String[] args){  
        Parent[] parents = new Parent[10]; //不会初始化类Parent
    }  
}
  • 常量在编译期间会存入调用类的常量池中,本质上并没有直接引用定义常量的类,不会触发定义常量所在的类。
class Const {
    static final int A = 100; //编译阶段,常量A存储到Init类的常量池中
    static {
        System.out.println("Const init");
    }
}
 
public class Init{  
    public static void main(String[] args){  
        System.out.println(Const.A);  
    }  
}

在编译阶段,Const类中常量A的值100存储到Init类的常量池中,这两个类在编译成class文件之后就没有联系了。

  • 通过类名获取Class对象,不会触发类的初始化。
public class test {
   public static void main(String[] args) throws ClassNotFoundException {
        Class c_dog = Dog.class; //不会初始化Dog类
        Class clazz = Class.forName("zzzzzz.Cat"); //会初始化Cat类
    }
}
 
class Cat {
    private String name;
    private int age;
    static {
        System.out.println("Cat is load");
    }
}
 
class Dog {
    private String name;
    private int age;
    static {
        System.out.println("Dog is load");
    }
}
  • 通过Class.forName加载指定类时,如果指定参数initialize为false时,也不会触发类初始化,其实这个参数是告诉虚拟机,是否要对类进行初始化

  • 通过ClassLoader默认的loadClass方法,也不会触发初始化动作;

相关文章

    暂无相关文章
相关栏目:

用户点评