首页 > 编程语言 >Java 类加载时机(动态、静态) 与 类加载过程 详解

Java 类加载时机(动态、静态) 与 类加载过程 详解

时间:2023-03-05 16:31:35浏览次数:45  
标签:初始化 Java 变量 静态 详解 阶段 public 加载

(目录)


类加载

动态加载 和 静态加载

反射机制是java实现动态语言的关键,也就是通过反射实现类动态加载

  1. 静态加载 : 编译时加载相关的类,如果没有则报错,依赖性太强
  2. 动态加载 : 运行时加载需要的类,如果运行时不用该类,则不报错,降低了依赖性

类加载时机

  1. 当创建对象时(new)——静态加载
  2. 当子类被加载时,父类也加载——静态加载
  3. 调用类中的静态成员时——静态加载
  4. 通过反射——动态加载

静态加载:

编译的时候就会加载这个类(检查类是否正确、依赖关系、父类是否ok等等),比如下面这段代码

image-20230305155905625

用户可能输入1,也可能输入2,Dog类不一定会用到,静态加载不管那么多,它在编译的时候就会加载这个类,并且进行语法的校验

image-20230305155915240

静态加载的依赖性较强


动态加载

执行到这里才加载

image-20230305155922918

因为new Dog()是静态加载,因此必须编写Dog Person类是动态加载,所以,没有编写Person类也不会报错,只有当动态加载该类时,才会报错


类加载过程

image-20230305155933969

加载和连接是JVM来完成的,我们无法控制,而初始化阶段就是我们自己可控的了(比如你在静态代码里面写什么内容是你自己可以控制的)

  • 加载:将类的class文件读入到内存中,并为之创建一个java.lang.Class对象,此过程由类加载器完成
  • 连接:将类的二进制数据合并到JRE中
  • 验证阶段:对文件进行安全性校验,比如文件格式是否正确、元数据验证是否正确、字节码是否正确、符号引用是否正确
  • 准备阶段给静态变量分配内存,给静态变量默认初始化
  • 解析阶段:虚拟机把常量池中的符号引用替换为直接引用
  • 初始化阶段静态代码块,静态变量显式赋值 初始化:JVM负责对类进行初始化,这里主要指静态成员

image-20230305155942385

  • 连接阶段-验证
  1. 目的是为了确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。
  2. 包括:文件格式验证(是否以魔数oxcafebabe开头)、元数据验证、字节码验证和符号引用验证
  3. 可以考虑使用-Xverify:none 参数来关闭大部分的类验证措施,缩短虚拟机类加载的时间。

连接阶段-准备

JVM会在该阶段对静态变量,分配内存并默认初始化(对应数据类型的默认初始值,如0、OL、null、false等)。这些变量所使用的内存都将在方法区中进行分配。

//类加载的连接阶段——准备
public class ClassLoad02 {
    public static void main(String[] args) {
        //...
    }
}

class A {
    //属性(成员变量,字段)
    //分析类加载的连接阶段——准备,属性是如何处理:
    //1. n1是成员变量,不是静态变量,因此在准备阶段,不会分配内存
    //2. n2是静态变量,分配内存,n2是默认初始化0,而不是20
    //3. n3是static final常量,和静态变量不一样,因为一旦赋值就不变,n3 = 30
    public int n1 = 10;
    public static int n2 = 20;
    public static final int n3 = 30;
}

连接阶段-解析

虚拟机将常量池内的符号引用替换为直接引用的过程。


Initialization(初始化)

  1. 到初始化阶段,才真正开始执行类中定义的Java程序代码,此阶段是执行<clinit>0方法的过程。
  2. <clinit>0方法是由编译器按语句在源文件中出现的顺序,依次自动收集类中的所有静态变量的赋值动作和静态代码块中的语句,并进行合并
  3. 虚拟机会保证一个类的<clinit>)方法在多线程环境中被正确地加锁、同步,如果多个线程同时去初始化一个类,那么只会有一个线程去执行这个类的<clinit>0方法,其他线程都需要阻塞等待,直到活动线程执行<clinit>0方法完毕[debug源码]
//类加载初始化阶段
public class ClassLoad03 {
    public static void main(String[] args) throws Exception {
        //1.加载B类,并生成对应的Class类对象
        //2.连接 num = 0;
        //3.初始化阶段:依次自动收集类中的所有静态变量的赋值动作和静态代码块中的语句,并合并
        /*
            clinit(){
                System.out.println("B 静态代码块被执行");
                //num = 300;
                num = 100;
            }
            合并:num = 100;
         */

        //new B(); //类加载
        //System.out.println(B.num); //100,如果直接使用类的静态属性,也会导致类的加载

        //加载类的时候,是有同步机制控制
        /*
            protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
                //正因为有这个机制,才能保证某个类在内存中,只有一个 Class 对象
                synchronized (getClassLoadingLock(name)) {
                    //...
                }
            }
         */
        B b = new B();
    }
}

class B {
    static {
        System.out.println("B 静态代码块被执行");
        num = 300;
    }

    static int num = 100;

    public B() {
        System.out.println("B 构造器被执行");
    }
}


标签:初始化,Java,变量,静态,详解,阶段,public,加载
From: https://blog.51cto.com/panyujie/6101464

相关文章

  • 三天吃透Java基础八股文
    本文已经收录到Github仓库,该仓库包含计算机基础、Java基础、多线程、JVM、数据库、Redis、Spring、Mybatis、SpringMVC、SpringBoot、分布式、微服务、设计模式、架构、校......
  • java内存分配
    1.栈方法运行时使用的内存。比如main方法运行,进入方法栈中执行2.堆存储对象或者数组。new来创建的都存储在堆内存3.方法区存储可以运行的class文件4.本地方法栈JVM......
  • java序列化 教程
    序列化的条件:序列化必须要满足两个条件:1.该对象必须实现java.io.Serializble接口2.改对象的所有属性必须是可序列化的,若有属性不可序列化也须注明是短暂的 注:检......
  • java8新特性-引用流ReferencePipeline
    ReferencePipeline实现了Stream接口,Stream接口定义了顺序和并行聚合的元素序列操作。publicinterfaceStream<T>extendsBaseStream<T,Stream<T>>{Stream<T......
  • 用javascript实现轮播图
    <!DOCTYPEhtml><htmllang="en"><head><metacharset="UTF-8"><metahttp-equiv="X-UA-Compatible"content="IE=edge"><metaname="viewport"conten......
  • java-spring aop
    导读AOP面向切面编程AspectorientedProgrammingOOP面向对象编程ObjectorientedProgramming作用:在不惊动原始设计的基础上进行功能增强。1、导入坐标<!--......
  • 加载内核模块-Unknown symbol错误分析
    在调试过程中,发现一个问题,编译生成一个KO文件,insmod加载后报错:Unknownsymbolvar_set_integer(err0)Unknownsymbolparse_arg_eq(err0)问题分析思路:一、用命令查......
  • JVM 类加载器、双亲委派原理
    类加载器ClassLoader作用:负责装入类。一个java程序运行,至少需要三个类加载器实例,负责加载不同类。BootstrapClassLoader是JVM内核内嵌加载器,主要负载加载JAVA_HOME/lib......
  • 以下总结了Java一些面试前准备和技术参考题,希望对你有所帮助
    以下真实模拟JAVA面试场景:感谢您参加我们的面试;以下是我们总结一些面试前准备和技术参考题,希望对你有所帮助。首先,你需要先准备下工作相关的自我介绍,包括以下内容:几年......
  • 学会了Java 8 Lambda表达式,简单而实用
    OneAPM摘要:此篇文章主要介绍Java8Lambda表达式产生的背景和用法,以及Lambda表达式与匿名类的不同等。本文系OneAPM工程师编译整理。Java是一流的面向对象语言,除了部分......