Java虚拟机工作原理

article/2025/8/22 15:35:32

首先我想从宏观上介绍一下Java虚拟机的工作原理。从最初的我们编写的Java源文件(.java文件)是如何一步步执行的,如下图所示,首先Java源文件经过前端编译器(javac或ECJ)将.java文件编译为Java字节码文件,然后JRE加载Java字节码文件,载入系统分配给JVM的内存区,然后执行引擎解释或编译类文件,再由即时编译器将字节码转化为机器码。主要介绍下图中的类加载器和运行时数据区两个部分。

  • 类加载

  类加载指将类的字节码文件(.class)中的二进制数据读入内存,将其放在运行时数据区的方法区内,然后在堆上创建java.lang.Class对象,封装类在方法区内的数据结构。类加载的最终产品是位于堆中的类对象,类对象封装了类在方法区内的数据结构,并且向JAVA程序提供了访问方法区内数据结构的接口。如下是类加载器的层次关系图。

    • 启动类加载器(BootstrapClassLoader):在JVM运行时被创建,负责加载存放在JDK安装目录下的jre\lib的类文件,或者被-Xbootclasspath参数指定的路径中,并且能被虚拟机识别的类库(如rt.jar,所有的java.*开头的类均被Bootstrap ClassLoader加载)。启动类无法被JAVA程序直接引用。
    • 扩展类加载器(Extension ClassLoader):该类加载器负责加载JDK安装目录下的\jre\lib\ext的类,或者由java.ext.dirs系统变量指定路径中的所有类库,开发者也可以直接使用扩展类加载器。
    • 应用程序类加载器(AppClassLoader):负责加载用户类路径(Classpath)所指定的类,开发者可以直接使用该类加载器,如果应用程序中没有定义过自己的类加载器,该类加载器为默认的类加载器。
    • 用户自定义类加载器(User ClassLoader):JVM自带的类加载器是从本地文件系统加载标准的java class文件,而自定义的类加载器可以做到在执行非置信代码之前,自动验证数字签名,动态地创建符合用户特定需要的定制化构建类,从特定的场所(数据库、网络中)取得java class。

注意如上的类加载器并不是通过继承的方式实现的,而是通过组合的方式实现的。而JAVA虚拟机的加载模式是一种委派模式,如上图中的1-7步所示。下层的加载器能够看到上层加载器中的类,反之则不行。类加载器可以加载类但是不能卸载类。说了一大堆,还是感觉需要拿点代码说事。

首先我们先定义自己的类加载器MyClassLoader,继承自ClassLoader,并覆盖了父类的findClass(String name)方法,如下:

 View Code

  我们如何利用我们定义的类加载器加载指定的字节码文件(.class)呢?如通过MyClassLoader加载C:\\Users\\Administrator\\下的Test.class字节码文件,代码如下所示:

复制代码

 
  1. public class Client {

  2. public static void main(String[] args) {

  3. // TODO Auto-generated method stub

  4. //MyClassLoader的父类加载器为系统默认的加载器AppClassLoader

  5. MyClassLoader myCLoader = new MyClassLoader("MyClassLoader");

  6. //指定MyClassLoader的父类加载器为ExtClassLoader

  7. //MyClassLoader myCLoader = new MyClassLoader(ClassLoader.getSystemClassLoader().getParent(),"MyClassLoader");

  8. myCLoader.setPath("C:\\Users\\Administrator\\");

  9. Class<?> clazz;

  10. try {

  11. clazz = myCLoader.loadClass("Test");

  12. Field[] filed = clazz.getFields(); //获取加载类的属性字段

  13. Method[] methods = clazz.getMethods(); //获取加载类的方法字段

  14. System.out.println("该类的类加载器为:" + clazz.getClassLoader());

  15. System.out.println("该类的类加载器的父类为:" + clazz.getClassLoader().getParent());

  16. System.out.println("该类的名称为:" + clazz.getName());

  17. } catch (ClassNotFoundException e) {

  18. // TODO Auto-generated catch block

  19. e.printStackTrace();

  20. }

  21. }

  22. }

复制代码

  • 运行时数据区

  字节码的加载第一步,其后分别是认证、准备、解析、初始化,那么这些步骤又具体做了哪些工作,以及他们会对运行时数据区缠身什么影响呢?如下图所示:

  如下我们将介绍运行时数据区,主要分为方法区、Java堆、虚拟机栈、本地方法栈、程序计数器。其中方法区和Java堆一样,是各个线程共享的内存区域,而虚拟机栈、本地方法栈、程序计数器是线程私有的内存区。

    1. Java堆:Java堆是Java虚拟机所管理的内存中最大的一块,被进程的所有线程共享,在虚拟机启动时被创建。该区域的唯一目的就是存放对象实例,几乎所有的对象实例都在这里分配内存,随着JIT编译器的发展与逃逸分支技术逐渐成熟,栈上分配、标量替换等优化技术使得对象在堆上的分配内存变得不是那么“绝对”。Java堆是垃圾收集器管理的主要区域。由于现在的收集器基本都采用分代收集算法,所以Java堆中还可以分为老年代和新生代(Eden、From Survivor、To Survivor)。根据Java虚拟机规范,Java堆可以处于物理上不连续的内存空间,只要逻辑上连续即可。该区域的大小可以通过-Xmx和-Xms参数来扩展,如果堆中没有内存完成实例分配,并且堆也无法扩展,将会抛出OutOfMemoryError异常。
    2. 方法区:用于存储被Java虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。不同于Java堆的是,Java虚拟机规范对方法区的限制非常宽松,可以选择不实现垃圾收集。但并非数据进入了方法区就“永久”存在了,这区域内存回收目标主要是针对常量池的回收和对类型的卸载。如果该区域内存不足也会抛出OutOfMemoryError异常。
      • 常量池:这个名词可能大家也经常见,它是方法区的一部分。Class文件除了有类的版本、字段、方法、接口等描述信息外,还有一项信息就是常量池,用于存放编译期生成的各种字面量和符号引用。Java虚拟机运行期间,也可能将新的常量放入常量池(如String类的intern()方法)。
    3. 虚拟机栈:线程私有,生命周期与线程相同。虚拟机栈描述的是Java方法执行的内存模型:每个方法在执行时都会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中入栈到出栈的过程。如果请求的站深度大于虚拟机所允许的深度,将抛出StackOverflowError异常,虚拟机栈在动态扩展时如果无法申请到足够的内存,就会抛出OutOfMemoryError异常。
    4. 本地方法栈:与虚拟机栈类似,不过虚拟机栈是为虚拟机执行Java方法(字节码)服务,而本地方法栈则是为虚拟机使用到的Native方法服务。该区域同样会报StackOverflowError和OutOfMemoryError异常。
    5. 程序计数器:一块较小的内存空间,可以看作是当前线程所执行的字节码的行号指示器。字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器完成。如果线程正在执行一个Java方法,计数器记录的是正在执行的虚拟机字节码指令的地址,如果正在执行的是Native方法,这个计数器值为空。此内存区域是唯一一个在Java虚拟机规范中没有规定任何OutOfMemoryError情况的区域。

  写了这么多,感觉还是少一个例子。通过最简单的一段代码解释一下,程序在运行时数据区个部分的变化情况。

复制代码

 
  1. public class Test{

  2. public static void main(String[] args){

  3. String name = "best.lei";

  4. sayHello(name);

  5. }

  6. public static void sayHello(String name){

  7. System.out.println("Hello " + name);

  8. }

  9. }

复制代码

  通过编译器将Test.java文件编译为Test.class,利用javap -verbose Test.class对编译后的字节码进行分析,如下图所示:

  我们在看看运行时数据区的变化:

本文转载自:https://www.cnblogs.com/zhanglei93/p/6590609.html


http://chatgpt.dhexx.cn/article/cH1BmOKi.shtml

相关文章

深入浅出Java虚拟机

深入浅出Java虚拟机 什么是Java虚拟机 虚拟机是一种抽象化的计算机&#xff0c;通过在实际的计算机上仿真模拟各种计算机功能来实现的。Java虚拟机有自己完善的硬体架构&#xff0c;如处理器、堆栈、寄存器等&#xff0c;还具有相应的指令系统。Java虚拟机屏蔽了与具体操作系统…

java虚拟机栈

栈 java指令是根据栈来设计的&#xff0c;优点就是跨平台&#xff0c;指令集小&#xff0c;编译容易实现&#xff0c;缺点是性能有所下降&#xff0c;实现同样的功能需要更多的指令 栈是运行时的的单位&#xff0c;堆是存储单位&#xff0c;栈管运行&#xff0c;堆管存储 ja…

深入Java虚拟机之类加载

深入Java虚拟机_ClassLoader 类加载器深入剖析 Java虚拟机与程序的生命周期 在如下几种情况下&#xff0c;Java虚拟机将结束生命周期 执行了System.exit()方法程序正常执行结束程序在执行过程中遇到了异常或错误而异常终止由于操作系统出现错误而导致Java虚拟机进程 类的加载…

Java虚拟机安装

下载VMware然后安装打开&#xff08;如果已经有了的就不要下载了&#xff09; 秘钥&#xff1a;FF31K-AHZD1-H8ETZ-8WWEZ-WUUVA CV7T2-6WY5Q-48EWP-ZXY7X-QGUWD 然后创建虚拟机 创建完成 然后开启虚拟机如果报以上错误&#xff1a; 重启计算机&#xff0c;进入bios&#xff08…

Java虚拟机面试题精选(一)

微信搜索【程序员囧辉】&#xff0c;关注这个坚持分享技术干货的程序员。 概述 现在面试Java开发时&#xff0c;基本都会问到Java虚拟机的知识&#xff0c;根据职位不同问的内容深浅又有所区别。本文整理了10道面试中常问的Java虚拟机面试题&#xff0c;希望对正在面试的同学有…

【JAVA虚拟机】java虚拟机

文章目录 前言一、JAVA内存结构体系二、双亲委派机制三、JAVA虚拟机调优场景1.CPU占用过高2.死锁3.内存泄漏4.元数据空间溢出性能调优归纳 三、垃圾回收器1、垃圾回收算法2、垃圾收集器 前言 记录java虚拟机常见的问题场景。 提示&#xff1a;以下是本篇文章正文内容&#xff…

java虚拟机是什么?

一、java虚拟机定义: 虚拟机是一种抽象化的计算机,通过在实际的计算机上仿真模拟各种计算机功能来实现的。Java虚拟机有自己完善的硬体架构,如处理器、堆栈、寄存器等,还具有相应的指令系统。Java虚拟机屏蔽了与具体操作系统平台相关的信息,使得Java程序只需生成在Java虚…

【Java虚拟机】万字长文,搞定Java虚拟机方方面面!

文章目录 1.JVM内存结构1.1.JVM内存结构图1.2.程序计数器1.3.虚拟机栈1.4.本地方法栈1.5.Java堆1.6.方法区1.7.StringTable1.8.直接内存 2.对象创建解析2.1.对象创建的流程2.2.对象的结构2.3.对象的访问方式 3.JVM垃圾回收3.1.垃圾回收概述3.2.引用计数法3.3.可达性分析算法3.4…

java虚拟机运行机制

首先简单阐述下解释型语言和编译型语言的联系与区别。 编译型语言是通过编译器将程序编译成目标机器所能识别的机器码&#xff0c;而解释型语言不需要编译过程。由该语言的解释器读取脚本&#xff0c;按照语法规则进行解释&#xff0c;然后调用解释器内建的命令(或者库函数)。例…

浅谈Java虚拟机(JVM)

前言&#x1f447; 第一次在CSDN上写博客&#xff08;2022.03.24&#xff0c;大二下&#xff09;&#xff0c;历时天&#xff0c;期间因为准备蓝桥杯有所延误。思考了一下&#xff0c;决定在第一篇写一写JVM&#xff0c;不能保证所有的东西都是对的&#xff0c;虚心求教&#x…

[JVM] Java虚拟机栈

1. 概念 Java虚拟机栈&#xff08;Java Virtual Machine Stacks&#xff09;是线程私有的&#xff0c;栈使用的内存不需要保证是连续的&#xff0c;栈帧存储了方法的局部变量表、操作数栈、动态连接和方法返回地址等信息。每一个方法从调用至执行完成的过程&#xff0c;都对应…

【JVM】Java虚拟机简介

【JVM】Java虚拟机简介 我们看中的并非Java语言&#xff0c;而是JVM。——Java之父James Gosling Java 虚拟机&#xff08;Java virtual machine&#xff0c;JVM&#xff09;是运行 Java 程序必不可少的机制。 JVM实现了Java语言最重要的特征&#xff1a;即平台无关性。这是因…

JAVA虚拟机概述

本博客内容为《深入理解Java虚拟机&#xff1a;JVM高级特性与最佳实践》的阅读笔记。 1 Java技术体系 仅从传统意义上来看&#xff0c;Sun官方所定义的Java技术体系包括以下几个组成部分&#xff1a; Java程序语言设计各种硬件平台上的Java虚拟机Class文件格式Java API类库来…

详解Java虚拟机

资料来源&#xff1a;尚硅谷宋红康JVM全套教程&#xff08;详解java虚拟机&#xff09;_哔哩哔哩_bilibili 1.JVM与Java体系结构 1.1. 前言 如果我们把核心类库的API比做数学公式的话&#xff0c;那么Java虚拟机的知识就好比公式的推导过程。 计算机系统体系对我们来说越来越远…

java虚拟机到底是什么

转自:http://blog.csdn.net/zhangjg_blog/article/details/20380971 http://blog.hesey.net/2011/04/introduction-to-java-virtual-machine.html 虚拟机是一种抽象化的计算机&#xff0c;通过在实际的计算机上仿真模拟各种计算机功能来实现的。Java虚拟机有自己完善的硬体架…

深入理解Java虚拟机到底是什么

什么是Java虚拟机 作为一个Java程序员&#xff0c;我们每天都在写Java代码&#xff0c;我们写的代码都是在一个叫做Java虚拟机的东西上执行的。但是如果要问什么是虚拟机&#xff0c;恐怕很多人就会模棱两可了。在本文中&#xff0c;我会写下我对虚拟机的理解。因为能力所限&am…

java虚拟机(JVM)

一、虚拟机的组成 虚拟机的组成主要有&#xff1a;方法区、堆都为线程共享区域&#xff0c;有线程安全问题。 栈、本地方法栈、程序计数器为线程的独享区域&#xff0c;不存在线程安全问题。 注&#xff1a;JVM的调优主要是针对堆和栈进行的 1.1&#xff1a;运行时数据区域 …

JVM——Java虚拟机架构

0. 前言 Java虚拟机&#xff08;Java virtualmachine&#xff09;实现了Java语言最重要的特征&#xff1a;即平台无关性。 平台无关性原理&#xff1a;编译后的 Java程序&#xff08;.class文件&#xff09;由 JVM执行。JVM屏蔽了与具体平台相关的信息&#xff0c;使程序可以…

深入理解Java虚拟机

前言 JVM是什么&#xff1f; JVM&#xff08;Java Virtual Machine&#xff0c;Java 虚拟机&#xff09;顾名思义就是用来执行 Java 程序的“虚拟主机”&#xff0c;实际的工作是将编译生成的.class 文件&#xff08;字节码&#xff09;翻译成底层操作系统可以运行的机器码并…

一文彻底了解JVM

目录 JVM内存结构 GC算法和收集器 JDK性能调优监控工具 GC调优 JVM内存结构 类加载机制 Java运行时编译源码(.java)成字节码&#xff0c;由jre运行。jre由java虚拟机实现。JVM分析字节码&#xff0c;后解释并执行。 1、全盘负责委托机制 当一个ClassLoader加载一个类的时…