JVM基础结构与字节码执行引擎

简介:

JVM基础结构与字节码执行引擎

JVM基础结构
JVM内部结构如下:栈、堆。


JVM中的栈主要是指线程里面的栈,里面有方法栈、native方法栈、PC寄存器等等;每个方法栈是由栈帧组成的;每个栈帧是由局部变量表、操作数栈等组成。

每个栈帧其实就代表一个方法


java中所有对象都在堆中分配;堆中对象又分为年轻代、老年代等等,不同代的对象使用不同垃圾回收算法。

-XMs:启动虚拟机预留的内存
-Xmx:最大的堆内存

一、堆的分代假设
根据研究表明,堆中对象大部分都是创建后,立马就可以被销毁的。如:

为了优化堆中的内存,将堆中对象分为不同代。在年轻代中,GC发生比较频繁;在老年代中,GC发生比较少。

二、堆的分代
年轻代:Young Generation
老年代:Old Generation/Tenured
永久代:Permanent Generation
永久代在Java虚拟机规范中是没有的,但是Host Spot虚拟机中有。

三、方法区
方法区被所有线程共享;方法区是用来存储编译后的代码,即存储每个类的运行时常量池、字段和方法。
方法区在虚拟机启动时创建;虽然方法区在逻辑上是堆的一部分,但在一些简单的实现中,方法区可以选择不进行垃圾回收和紧凑化。

方法区在java8的变化
java7之前:方法区的实现:永久代,是作为堆的一部分;
java8之后:方法区的实现:metaspace,是堆外的内存;
1、为什么要这样改变?
因为java可以动态加载字节码信息,这样方法区就会慢慢的挤占堆中内存。为了避免与堆争抢内存,java8将方法区的实现移至堆外。
2、方法区、永久代、MetaSpace的区别?
方法区是java虚拟机规范所规定的一个概念。其中java7实现方法区的地方称为永久代;java8实现方法区的地方称为MetaSpace

字节码文件的结构
java程序在运行的时候,将源码编译成字节码,字节码在不同系统上的JVM翻译成对应的机器码。这是Java平台无关性的基础。

但是,编译后的字节码是如何读取到JVM中的?字节码执行引擎是如何识别、执行指令?

1、如何查看字节码文件

classpy工具
IDEA的jclasslib Bytecode viewer插件
2、字节码文件结构
一个字节码文件包含以下部分:

(1)magic:0xCAFEBABE
class文件的magic code,用于标识该文件是class文件。

(2)minor_version、major_version
用于标识该class文件的版本,防止高版本的class文件被低版本的JVM读取并执行。

(3)constant_pool:常量池
用于存储该class文件经常被使用的信息,优化内存。比如说System.out.print()

(4)access_flag
表示这个类得访问权限,对应到java源码就是public、final之类的

字节码执行引擎
这里以一个线程为例。一般来说,一个方法栈最底层的栈帧都是Thread.run方法。当一个线程准备调用另一个方法时,会先将实参拷贝一份到新栈帧的局部变量表里,然后再执行代码。
1、局部变量表
每次调用新方法时,会默认将当前对象的地址this作为局部变量表的第一个参数;后面存放传过来的参数。这与javascript的做法很相似。

2、方法调用的相关指令

invokevirtual:一般实例方法,有多态;
invokeinterface:接口方法,有多态
invokestatuc:静态方法,无多态
invokespecial:特殊方法,无多态
invokedynamic:动态调用,JDK7新增,方法无需在编译时确定
3、方法调用的过程
(1)在开始时

方法栈新增一个栈帧;
实例方法的this、参数放到局部变量表中;
开始新栈帧中字节码的执行;
(2)在返回时

将返回值放在调用者方法栈帧中的操作数栈上;
(3)在异常出现时

寻找匹配的异常处理代码
(4)在finally时

为每个分支新增一个跳转
4、为什么Mockito、EasyMock无法对private、static方法进行mock?
因为他们mock方法是通过覆盖这些方法来实现的,而private、static没法被覆盖。PowerMock是通过修改字节码文件达到mock私有、静态方法的。

原博客地址

转载地址https://www.cnblogs.com/fourther/p/12673213.html

相关文章
|
4月前
|
Java
JVM运行时数据区(内存结构)
1)虚拟机栈:每次调用方法都会在虚拟机栈中产生一个栈帧,每个栈帧中都有方法的参数、局部变量、方法出口等信息,方法执行完毕后释放栈帧 (2)本地方法栈:为native修饰的本地方法提供的空间,在HotSpot中与虚拟机合二为一 (3)程序计数器:保存指令执行的地址,方便线程切回后能继续执行代码
45 3
|
5月前
|
存储 算法 Java
聊聊jvm的内存结构, 以及各种结构的作用
【10月更文挑战第27天】JVM(Java虚拟机)的内存结构主要包括程序计数器、Java虚拟机栈、本地方法栈、Java堆、方法区和运行时常量池。各部分协同工作,为Java程序提供高效稳定的内存管理和运行环境,确保程序的正常执行、数据存储和资源利用。
135 10
|
5月前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
95 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
|
7月前
|
存储 算法 Java
JVM组成结构详解:类加载、运行时数据区、执行引擎与垃圾收集器的协同工作
【8月更文挑战第25天】Java虚拟机(JVM)是Java平台的核心,它使Java程序能在任何支持JVM的平台上运行。JVM包含复杂的结构,如类加载子系统、运行时数据区、执行引擎、本地库接口和垃圾收集器。例如,当运行含有第三方库的程序时,类加载子系统会加载必要的.class文件;运行时数据区管理程序数据,如对象实例存储在堆中;执行引擎执行字节码;本地库接口允许Java调用本地应用程序;垃圾收集器则负责清理不再使用的对象,防止内存泄漏。这些组件协同工作,确保了Java程序的高效运行。
70 3
|
7月前
|
存储 安全 Java
JVM内存结构
这篇文章详细介绍了Java虚拟机(JVM)的内存结构,包括类的加载过程、类加载器的双亲委派机制、沙箱安全机制、程序计数器、Java栈、Java堆、本地方法和本地方法栈等关键组件及其作用。
JVM内存结构
|
7月前
|
C# UED 开发者
WPF动画大揭秘:掌握动画技巧,让你的界面动起来,告别枯燥与乏味!
【8月更文挑战第31天】在WPF应用开发中,动画能显著提升用户体验,使其更加生动有趣。本文将介绍WPF动画的基础知识和实现方法,包括平移、缩放、旋转等常见类型,并通过示例代码展示如何使用`DoubleAnimation`创建平移动画。此外,还将介绍动画触发器的使用,帮助开发者更好地控制动画效果,提升应用的吸引力。
373 0
|
8月前
|
存储 缓存 自然语言处理
(三)JVM成神路之全面详解执行引擎子系统、JIT即时编译原理与分派实现
执行引擎子系统是JVM的重要组成部分之一,在JVM系列的开篇曾提到:JVM是一个架构在平台上的平台,虚拟机是一个相似于“物理机”的概念,与物理机一样,都具备代码执行的能力。
151 1
|
8月前
|
存储 运维 Java
Java面试题:JVM的内存结构有哪些主要部分?请简述每个部分的作用
Java面试题:JVM的内存结构有哪些主要部分?请简述每个部分的作用
87 9
|
7月前
|
存储 监控 算法
深入解析JVM内部结构及GC机制的实战应用
深入解析JVM内部结构及GC机制的实战应用
|
8月前
|
存储 设计模式 监控
Java面试题:简述JVM的内存结构,包括堆、栈、方法区等。栈内存优化的方法有 哪些?
Java面试题:简述JVM的内存结构,包括堆、栈、方法区等。栈内存优化的方法有 哪些?
66 0