吊打面试官(一)-Java程序执行流程详细分析

createh52周前 (03-04)技术教程2

一个Java程序是如何执行的呢,这个见鬼的问题可以很简单,也可以很复杂。

如果你回答点一下run就执行了,相信面试官会把你当一只鬼。

如果你按照下面这样描述,相信面试官会把你当做一个神。


程序例子如下:


class Rectangle {
int width;
int height;

Rectangle(int w, int h) {
width = w;
height = h;
}

int getArea() {
return width * height;
}
}

public class Main {
public static void main(String[] args) {
Rectangle rect = new Rectangle(5, 10);
int area = rect.getArea();

if (area > 50) {
System.out.println("面积大于 50");
} else {
System.out.println("面积小于或等于 50");
}
}
}


按照执行时间顺序,程序经历了如下步骤:


1.程序启动:

从用户输入命令开始,操作系统接收请求并为Java进程分配资源,包括虚拟内存和创建进程控制块。JVM启动引导程序,加载基础Java类,初始化JVM内存区域,并启动类加载器。


操作系统步骤:

? 用户在命令行输入`java Main`或通过 IDE 触发程序运行。

? 操作系统内核接收启动新进程的请求。

? 为 Java 进程分配一个独立的虚拟内存空间,大小根据进程需求和系统配置而定。

? 初始化进程控制块(PCB),其中包含进程的状态、优先级、程序计数器等信息。

? 从磁盘加载 Java 可执行文件(`java.exe`或`java`命令对应的二进制文件)到内存。

? 将 CPU 的执行权交给 Java 进程,开始执行入口点代码。


JVM 处理步骤:

? JVM 启动引导程序(Bootstrap ClassLoader),从核心类库路径加载基础的 Java 类,如`java.lang.Object`。

? 初始化 JVM 内存区域:

? 方法区:分配内存用于存储类的元数据,包括类的结构信息、常量池、字段和方法数据等。

? 堆:为对象实例分配内存的区域,初始大小根据配置而定,后续可根据需要进行扩展。

? 栈:每个线程都有一个独立的栈,用于存储方法调用和局部变量。初始化主栈,为`main`方法准备栈帧。

? 程序计数器:设置为指向`main`方法的第一条指令的地址。

? 本地方法栈:用于执行本地方法(Native Method)。

? 启动类加载器,准备加载应用程序的类。


2.加载类

JVM通过类加载器加载需要的类,如Rectangle类和Main类,进行字节码验证,存储元数据信息,并为这些类分配内存。


JVM 处理步骤:

? 类加载器按照双亲委派模型,首先尝试从父类加载器加载`Rectangle`类。

? 若父类加载器未找到,则当前类加载器读取`Rectangle.class`字节码文件到内存缓冲区。

? 对字节码进行验证:

? 检查魔数(特定的文件标识)是否正确。

? 验证版本号是否符合当前 JVM 要求。

? 检查常量池中的常量是否合法。

? 验证字段和方法表的结构和访问权限。 ? 将类的元数据信息存储到方法区:

? 类名、父类名、接口列表。

? 字段信息:名称、类型、访问修饰符等。

? 方法信息:名称、参数列表、返回类型、字节码指令等。

? 对`Main`类进行同样的加载和验证过程。


内存处理步骤:

? 在方法区中为`Rectangle`类和`Main`类分别分配一块连续的内存空间,大小根据类的元数据大小而定。


3.执行`main`方法

创建新的栈帧以执行main方法,初始化局部变量,指向第一条指令,并开始执行方法体内的代码。


JVM 处理步骤:

? 在 Java 栈中为`main`方法创建一个新的栈帧。

? 为局部变量表分配空间,初始化局部变量`rect`和`area`为默认值(`null`和 0)。

? 将程序计数器设置为指向`main`方法的第一条指令的地址。


CPU 执行步骤:

? CPU 从内存中读取`new Rectangle(5, 10)`指令到指令寄存器。

? 控制单元解码指令,确定操作类型为对象创建和构造函数调用,并获取操作数(5 和 10)。? 执行`new`指令的操作:

? 计算对象所需内存大小,在堆中分配内存空间。

? 初始化对象的头部信息,包括类的元数据指针、锁状态等。


4.创建`Rectangle`对象

当遇到new关键字时,JVM会为新对象分配内存空间,在堆中存储字段值和对象头信息,并执行构造函数。


JVM 处理步骤:

? 执行`new`指令,在堆中为新对象分配足够的内存空间来存储字段`width`、`height`和对象头信息。

? 将对象的引用地址存储到局部变量`rect`中。

? 调用构造函数`Rectangle(int w, int h)`,将参数值传递给对应的字段。


内存处理步骤:

? 在堆内存中按照对象的结构存储字段值,`width`为 5,`height`为 10,同时存储对象头信息(如类的元数据指针、锁状态等)。


CPU 执行步骤:

? 执行赋值指令,将参数 5 写入堆中对象`width`字段对应的内存位置。

? 执行赋值指令,将参数 10 写入堆中对象`height`字段对应的内存位置。


5.调用`getArea`方法

对于像`getArea`这样的实例方法,JVM会创建新的栈帧,执行方法体中的指令,并将结果返回给调用者。


JVM 处理步骤:

? 在栈中为`getArea`方法创建新的栈帧,分配局部变量表空间。

? 将`this`引用(指向刚创建的`Rectangle`对象)压入栈帧的局部变量表。

? 将程序计数器设置为指向`getArea`方法的第一条指令。


CPU 执行步骤:

? CPU 从内存中读取`getArea`方法的字节码指令到指令寄存器。

? 解码并执行乘法运算指令,将`width`和`height`字段的值相乘,得到面积值。

? 将结果存储到局部变量`area`中,并将值返回给`main`方法的栈帧。


6.条件判断执行

根据计算的结果进行条件判断,通过CPU指令执行。


CPU 执行步骤:

? CPU 从内存中读取比较指令`area > 50`及其操作数到指令寄存器。

? 解码比较指令,执行比较操作,将`area`的值(50)与 50 进行比较。

? 根据比较结果设置相应的标志位(如零标志位、符号标志位等)。


7.输出相应结果


通过系统调用输出信息到控制台。JVM 处理步骤:? 根据条件判断的结果,确定要执行的`System.out.println`语句。? 查找`System.out`对象的引用,并调用其`println`方法。操作系统步骤:? JVM 向操作系统发起系统调用请求,传递要输出的字符串参数。? 操作系统接收请求,将字符串传递给终端驱动程序,最终显示在控制台窗口。CPU 执行步骤:? 执行与系统调用相关的指令,包括参数传递、中断处理等。


8.程序结束

当`main`方法所有指令执行完毕后,JVM会弹出栈帧,进行垃圾回收,销毁类加载器及相关资源,而操作系统则负责回收进程占用的资源,并最终终止进程。


JVM 处理步骤:

? 执行完`main`方法的所有指令,弹出`main`方法的栈帧。

? 检查堆中是否有未被引用的对象,进行垃圾回收,释放内存。

? 销毁类加载器和相关资源。


操作系统步骤:

? 回收 Java 进程占用的全部内存空间、关闭文件描述符等资源。

? 将进程状态设置为终止,并从进程列表中移除。



╰(*′︶`*)╯

相关文章

JVM类加载机制和java程序执行流程

JVM类加载机制与过程1. 类加载的基本概念类加载:指将.class文件中的字节码装载到Java虚拟机(JVM)中,以便后续的链接、初始化和执行。类加载器(ClassLoader):负责实际加载类的组...

探秘 Java Class 类文件:结构与原理深度解析

在 Java 编程的世界里,class 文件是 Java 程序运行的基石。深入理解 class 文件的结构与原理,不仅能帮助开发者更好地掌握 Java 虚拟机(JVM)的运行机制,还能为优化程序性能、...

体育老师教你学Java语言(上篇)

♂? 第一章:编程基础热身操1. Java语言简介 → 如何选择适合自己的运动项目Java的诞生:就像运动项目的选择需要考虑个人体质,Java诞生于1995年,最初叫Oak,后因类似咖啡的发音更名为J...

用JAVA代码写一个JAVA代码文本

这是可以通过 Java 程序来实现的。比如可以用 Java 代码往文件中写入一些 Java 代码内容,以下是一个简单示例,它会生成一个简单的 Java 类代码并写入到文件中:import java.i...

Java项目本地部署宝塔搭建实战java外卖小程序源码

大家好啊,我是测评君,欢迎来到web测评。本期给大家带来一套java开发的外卖小程序源码,这套系统已经完成了线下配送的大部分功能,适合学习与二次开发。技术架构技术框架:springboot + ssm...