jvm 类的加载原理(详细)

简单过程:源文件经过编译成二进制字节码文件保存到磁盘,jvm读取字节码文件经过类加载器加载,然后经过解释器解释成各个系统能够读懂执行二进制机器码。

一,编译:

Java源文件(.java)通过JDK的java编译器(javac.exe)编译成二进制字节码文件(.class)保存到硬盘。(以maven项目为例,经过编译后产生jar包或者war包保存到target目录下)

编译后的字节码文件格式主要分为两部分:常量池和方法字节码。

  常量池:记录的是代码出现过的字面量(文本字符串、八种基本类型的值、被声明为final的常量等)以及符号引用(类和方法的全限定名、字段的名称和描述符、方法的名称和描述符);

 https://www.cnblogs.com/superyc/p/9975254.html

方法字节码:  中放的是各个方法的字节码(依赖操作数栈和局部变量表,由JVM解释执行)

二,运行:

1,类的加载

(加载->连接->初始化)

连接包括(验证->准备->解析)

 

加载

1, 通过全类名获取定义此类的二进制字节流(从zip,jar,ear,war等获取)

2,将字节流所代表的静态存储结构转换为方法区的运行时数据结构

 

3,在内存中生成一个代表该类的 Class 对象(java.Lang.Class),作为方法区这些数据的访问入口

 

准备

准备阶段是正式为类变量分配内存并设置类变量初始值的阶段,这些内存都将在方法区中分配。

  1. 这时候进行内存分配的仅包括类变量(static),而不包括实例变量,实例变量会在对象实例化时随着对象一块分配在 Java 堆中。
  2. 这里所设置的初始值"通常情况"下是数据类型默认的零值(如0、0L、null、false等),比如我们定义了public static int value=111 ,那么 value 变量在准备阶段的初始值就是 0 而不是111(初始化阶段才会复制)。特殊情况:比如给 value 变量加上了 fianl 关键字public static final int value=111 ,那么准备阶段 value 的值就被复制为 111。

解析

解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。解析动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用限定符7类符号引用进行。

初始化

初始化是类加载的最后一步,也是真正执行类中定义的 Java 程序代码(字节码),初始化阶段是执行类构造器 <clinit> ()方法的过程。(通俗点就是初始化实例对象,给对象分配内存)

public class Main {
 
    public static void main(String[] args) {
        Animal animal = new Animal("Tom");
        animal.printName();
    }
 
}
 
class Animal{
    private String name;
 
    public Animal(String name) {
        super();
        this.name = name;
    }
     
    public void printName(){
        System.out.println("Animal = " + this.name);
    }
}

具体实例(Test.java表示当前类)(Anmimal表示class类对象)(animal 表示类的对象,实例)

(1)在类路径下找到编译好的 java 程序中得到 Test.class 字节码文件后,在命令行上敲 java Test,系统就会启动一个 JVM 进程,JVM进程从classpath路径下找到一个名为Test.class的二进制文件,将Test.class文件中的 类信息加载到运行时数据区的方法区(JDK 8 方法区存在 堆区) 中,这一过程叫做类的加载。(加载)(只有类信息在方法区中,才能创建对象,使用类中的成员变量);

(2)JVM 找到main方法的主函数入口, 持有一个指向当前类(Test)常量池的指针,而常量池中的第一项发现是一个对Animal对象的符号引用,并且main方法中第一条指令是Animal animal = new Animal("Tom"),就是让JVM创建一个Animal类的实例,但是方法区中还没有Animal类的类信息,于是JVM就要马上加载编译好Animal.class,将Animal类信息放入到方法区中,并且产生了一个Anmimal类对象,于是JVM 以一个直接指向方法区 Anmimal类对象的指针(直接引用)替换了常量池中第一项的符号引用。解析

(3)加载完Animal类的信息以后,JVM虚拟机就会在堆内存中为一个Anmimal类对象的实例animal分配内存,然后调用其构造函数初始化animal实例,(初始化)这个实例持有指向方法区的Anmimal类对象的类型信息(其中包含有方法表,java动态绑定的底层实现)的引用。(animal指向了Anmimal类对象的引用会自动的放在栈中,字符串常量"Tom"会自动的放在方法区的运行时常量池中,对象会自动的放入堆区)

(4)当使用 animal.pringName()的时候,JVM根据栈中animal引用找到Anmimal类对象,然后根据Animal类对象持有的引用定位到方法区中Animal类的类型信息方法表,获得pringName()函数的字节码地址,然后Java虚拟机执行引擎依赖局部变量表,操作数栈进行字节码解释执行,(解释执行)返回结果!

 

 

 

 

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值