JVM运行时数据区域-附面试题
Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。这些区域
有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而一直存在,有些区域则是
依赖用户线程的启动和结束而建立和销毁。
1. 程序计数器(Program Counter Register)
- 作用:程序计数器是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器。字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器来完成。(简化:用来存储指向下一条指令的地址,即将要执行的指令代码)
- 线程私有:由于 Java 虚拟机的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的,在任何一个确定的时刻,一个处理器(对于多核处理器来说是一个内核)都只会执行一条线程中的指令。因此,为了线程切换后能恢复到正确的执行位置,每条线程都需要有一个独立的程序计数器,各条线程之间计数器互不影响,独立存储。
- 无内存溢出情况:程序计数器是唯一一个在《Java 虚拟机规范》中没有规定任何 OutOfMemoryError 情况的区域。
2. Java 虚拟机栈(Java Virtual Machine Stacks)
- 作用:与程序计数器一样,Java 虚拟机栈也是线程私有的,它的生命周期与线程相同。虚拟机栈描述的是 Java 方法执行的线程内存模型:每个方法被执行的时候,Java 虚拟机都会同步创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态连接、方法出口等信息。每一个方法被调用直至执行完毕的过程,就对应着一个栈帧在虚拟机栈中从入栈到出栈的过程。
- 局部变量表:存放了编译期可知的各种 Java 虚拟机基本数据类型(boolean、byte、char、short、int、float、long、double)、对象引用(reference 类型,它并不等同于对象本身,可能是一个指向对象起始地址的引用指针,也可能是指向一个代表对象的句柄或者其他与此对象相关的位置)和 returnAddress 类型(指向了一条字节码指令的地址)。
- 异常情况:如果线程请求的栈深度大于虚拟机所允许的深度,将抛出 StackOverflowError 异常;如果 Java 虚拟机栈容量可以动态扩展,当栈扩展时无法申请到足够的内存会抛出 OutOfMemoryError 异常。
3. 本地方法栈(Native Method Stacks)
- 作用:本地方法栈与虚拟机栈所发挥的作用是非常相似的,其区别只是虚拟机栈为虚拟机执行 Java 方法(也就是字节码)服务,而本地方法栈则是为虚拟机使用到的本地(Native)方法服务。
- 本地方法是使用非 Java 语言(如 C、C++)实现的方法,它们可以直接访问底层操作系统的资源。例如:Java 程序有时需要与操作系统的底层功能进行交互,比如文件操作、网络操作等。由于 Java 本身的跨平台性,有些底层操作无法直接实现,这时就需要借助本地方法。
- 异常情况:和虚拟机栈一样,本地方法栈也会在栈深度溢出或者栈扩展失败时分别抛出 StackOverflowError 和 OutOfMemoryError 异常。
4. Java 堆(Java Heap)
- 作用:Java 堆是被所有线程共享的一块内存区域,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例都在这里分配内存。从内存分配的角度来看,线程共享的 Java 堆中可能划分出多个线程私有的分配缓冲区(Thread Local Allocation Buffer,TLAB)。不过无论如何划分,都与存放内容无关,无论哪个区域,存储的都仍然是对象实例。
- 垃圾回收的主要区域:Java 堆是垃圾收集器管理的主要区域,因此很多时候也被称作 “GC 堆”。根据垃圾收集算法的不同,Java 堆还可以细分为:新生代和老年代;再细致一点有 Eden 空间、From Survivor 空间、To Survivor 空间等。
- 异常情况:如果在 Java 堆中没有内存完成实例分配,并且堆也无法再扩展时,Java 虚拟机将会抛出 OutOfMemoryError 异常。
5. 方法区(Method Area)
- 作用:方法区是存放基础信息的位置,线程共享。主要包含三部分内容:1.类的元信息,保存了所有类的基本信息,2.运行时常量池,保存了字节码文件中的常量池内容、3.字符串常量池,保存了字符串常量
- 运行时常量池:是方法区的一部分,Class 文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池表(Constant Pool Table),用于存放编译期生成的各种字面量与符号引用,这部分内容将在类加载后存放到方法区的运行时常量池中。运行时常量池相对于 Class 文件常量池的另外一个重要特征是具备动态性,Java 语言并不要求常量一定只有编译期才能产生,也就是说,并非预置入 Class 文件中常量池的内容才能进入方法区运行时常量池,运行期间也可以将新的常量放入池中,这种特性被开发人员利用得比较多的便是 String 类的 intern () 方法。
- 异常情况:当方法区无法满足新的内存分配需求时,将抛出 OutOfMemoryError 异常。
- 方法区的实现
-
- JDK7及之前的版本将方法区存放在堆区域中的永久代空间,堆的大小由虚拟机参数来控制。
- JDK8及之后的版本将方法区存放在元空间中,元空间位于操作系统维护的直接内存中,默认情况下只要不超过操作系统承受的上限,可以一直分配。
6. 直接内存(Direct Memory)
- 作用:直接内存并不是虚拟机运行时数据区的一部分,也不是《Java 虚拟机规范》中定义的内存区域。但是这部分内存也被频繁地使用,而且也可能导致 OutOfMemoryError 异常出现。在 JDK 1.4 中新加入了 NIO(New Input/Output)类,引入了一种基于通道(Channel)与缓冲区(Buffer)的 I/O 方式,它可以使用 Native 函数库直接分配堆外内存,然后通过一个存储在 Java 堆里面的 DirectByteBuffer 对象作为这块内存的引用进行操作。这样能在一些场景中显著提高性能,因为避免了在 Java 堆和 Native 堆中来回复制数据。
- 异常情况:由于直接内存不受 Java 堆大小的限制,但是受本机总内存(包括 RAM 及 SWAP 区或者分页文件)大小以及处理器寻址空间的限制,所以在动态分配直接内存时可能会出现 OutOfMemoryError 异常。
面试题
基础概念类
- 问题:请简要介绍一下 JVM 运行时数据区域包含哪些部分?
参考答案:JVM 运行时数据区域主要包含以下几个部分:
-
- 程序计数器:线程私有,可看作当前线程所执行的字节码的行号指示器,是唯一不会出现 OutOfMemoryError 的区域。
- Java 虚拟机栈:线程私有,描述 Java 方法执行的线程内存模型,每个方法执行会创建栈帧,存储局部变量表、操作数栈等信息,可能抛出 StackOverflowError 和 OutOfMemoryError 异常。
- 本地方法栈:与虚拟机栈类似,为虚拟机使用的本地方法服务,也可能抛出 StackOverflowError 和 OutOfMemoryError 异常。
- Java 堆:线程共享,是对象实例分配内存的主要区域,也是垃圾收集器管理的主要区域,可能抛出 OutOfMemoryError 异常。
- 方法区:线程共享,用于存储已被虚拟机加载的类型信息、常量、静态变量等数据,可能抛出 OutOfMemoryError 异常。
- 运行时常量池:是方法区的一部分,存放编译期生成的字面量与符号引用,具备动态性。
- 直接内存:不是虚拟机运行时数据区的一部分,但也会被频繁使用,可能导致 OutOfMemoryError 异常。
- 问题:哪些区域是线程共享的,哪些是线程私有的?
参考答案:线程共享的区域有 Java 堆、方法区(包含运行时常量池);线程私有的区域有程序计数器、Java 虚拟机栈、本地方法栈。
原理机制类
- 问题:Java 虚拟机栈中栈帧的作用是什么,包含哪些内容?
参考答案:栈帧是 Java 虚拟机栈中用于支持方法调用和方法执行的数据结构。每一个方法从调用开始到执行完成的过程,都对应着一个栈帧在虚拟机栈中入栈到出栈的过程。栈帧包含以下内容:
-
- 局部变量表:用于存储方法参数和方法内部定义的局部变量。
- 操作数栈:在方法执行过程中,用于存储中间计算结果和临时数据。
- 动态连接:将符号引用转换为直接引用,实现方法的动态绑定。
- 方法出口:记录方法执行完毕后,从哪个位置继续执行调用该方法的后续代码。
- 问题:运行时常量池的动态性体现在哪里?
参考答案:运行时常量池的动态性体现在它并不局限于编译期生成的常量。Java 语言允许在运行期间将新的常量放入池中,例如 String 类的 intern () 方法。当调用 intern () 方法时,如果运行时常量池中已经包含一个等于此 String 对象的字符串,则返回常量池中的字符串;否则,将此 String 对象添加到常量池中,并返回该 String 对象的引用。
异常处理类
- 问题:在 JVM 运行时数据区域中,哪些区域可能会抛出 OutOfMemoryError 异常?
参考答案:可能抛出 OutOfMemoryError 异常的区域有 Java 堆、方法区、Java 虚拟机栈(当栈容量可以动态扩展时)、本地方法栈(当栈容量可以动态扩展时)和直接内存。例如,当 Java 堆中没有足够的内存来分配新的对象实例,并且堆也无法再扩展时,会抛出 OutOfMemoryError 异常;方法区无法满足新的内存分配需求时,也会抛出该异常。 - 问题:StackOverflowError 和 OutOfMemoryError 有什么区别?
参考答案:StackOverflowError 通常是由于线程请求的栈深度大于虚拟机所允许的深度而抛出的异常,一般是在递归调用方法时没有正确的终止条件,导致栈帧不断入栈,最终栈空间耗尽。而 OutOfMemoryError 是在无法申请到足够的内存时抛出的异常,它可能发生在 Java 堆、方法区、虚拟机栈(动态扩展时)、本地方法栈(动态扩展时)和直接内存等区域。
应用场景类
- 问题:在实际开发中,如何优化 JVM 运行时数据区域的使用?
参考答案:可以从以下几个方面进行优化:
-
- 对于 Java 堆:合理设置堆的大小,避免堆空间过大导致内存浪费或过小导致频繁的垃圾回收。可以根据应用程序的特点,调整新生代和老年代的比例。
- 对于方法区:避免加载过多不必要的类,及时卸载不再使用的类。可以通过设置合适的方法区大小,避免方法区内存溢出。
- 对于 Java 虚拟机栈:合理控制方法调用的深度,避免递归调用过深导致 StackOverflowError。可以适当调整栈的大小。
- 对于直接内存:合理使用 DirectByteBuffer,避免过度分配直接内存。在使用完后,及时释放直接内存资源。
- 问题:请举例说明 JVM 运行时数据区域在多线程环境下的应用和可能遇到的问题。
参考答案:在多线程环境下,每个线程都有自己独立的程序计数器、Java 虚拟机栈和本地方法栈,这些线程私有区域保证了线程之间的独立性。例如,多个线程同时执行不同的方法时,每个线程的栈帧操作互不干扰。而 Java 堆和方法区是线程共享的,多个线程可能会同时访问和修改堆中的对象和方法区中的类信息。可能遇到的问题包括:
-
- 线程安全问题:多个线程同时访问和修改堆中的对象时,可能会导致数据不一致的问题,需要使用同步机制来保证线程安全。
- 内存泄漏问题:如果线程持有对堆中对象的引用,而这些对象不再使用,但线程没有及时释放这些引用,可能会导致内存泄漏。
- 竞争问题:多个线程同时竞争方法区中的类加载锁时,可能会导致性能下降。
相关文章:

JVM运行时数据区域-附面试题
Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。这些区域 有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而一直存在,有些区域则是 依赖用户线程的启动和结束而建立和销毁。 1. 程序计…...

增删改查(CRUD)操作
文章目录 MySQL系列:1.CRUD简介2.Create(创建)2.1单行数据全列插入2.2 单行数据指定插入2.3 多⾏数据指定列插⼊ 3.Retrieve(读取)3.1 Select查询3.1.1 全列查询3.1.2 指定列查询3.1.3 查询字段为表达式(都是临时表不会对原有表数据产生影响)…...
Vue.js `Suspense` 和异步组件加载
Vue.js Suspense 和异步组件加载 今天我们来聊聊 Vue 3 中的一个强大特性:<Suspense> 组件,以及它如何帮助我们更优雅地处理异步组件加载。如果你曾在 Vue 项目中处理过异步组件加载,那么这篇文章将为你介绍一种更简洁高效的方式。 什…...

HTB:LinkVortex[WriteUP]
目录 连接至HTB服务器并启动靶机 信息收集 使用rustscan对靶机TCP端口进行开放扫描 使用nmap对靶机TCP开放端口进行脚本、服务扫描 使用nmap对靶机TCP开放端口进行漏洞、系统扫描 使用nmap对靶机常用UDP端口进行开放扫描 使用gobuster对靶机进行路径FUZZ 使用ffuf堆靶机…...

Linux命令入门
Linux命令入门 ls命令 ls命令的作用是列出目录下的内容,语法细节如下: 1s[-a -l -h] [Linux路径] -a -l -h是可选的选项 Linux路径是此命令可选的参数 当不使用选项和参数,直接使用ls命令本体,表示:以平铺形式,列出当前工作目录下的内容 ls命令的选项 -a -a选项&a…...

【问题】Chrome安装不受支持的扩展 解决方案
此扩展程序已停用,因为它已不再受支持 Chromium 建议您移除它。详细了解受支持的扩展程序 此扩展程序已停用,因为它已不再受支持 详情移除 解决 1. 解压扩展 2.打开manifest.json 3.修改版本 将 manifest_version 改为3及以上 {"manifest_ver…...

【题解】AtCoder Beginner Contest ABC391 D Gravity
题目大意 原题面链接 在一个 1 0 9 W 10^9\times W 109W 的平面里有 N N N 个方块。我们用 ( x , y ) (x,y) (x,y) 表示第 x x x 列从下往上数的 y y y 个位置。第 i i i 个方块的位置是 ( x i , y i ) (x_i,y_i) (xi,yi)。现在执行无数次操作,每一次…...

使用 SpringBoot+Thymeleaf 模板引擎进行 Web 开发
目录 一、什么是 Thymeleaf 模板引擎 二、Thymeleaf 模板引擎的 Maven 坐标 三、配置 Thymeleaf 四、访问页面 五、访问静态资源 六、Thymeleaf 使用示例 七、Thymeleaf 常用属性 前言 在现代 Web 开发中,模板引擎被广泛用于将动态内容渲染到静态页面中。Thy…...
【Java异步编程】CompletableFuture综合实战:泡茶喝水与复杂的异步调用
文章目录 一. 两个异步任务的合并:泡茶喝水二. 复杂的异步调用:结果依赖,以及异步执行调用等 一. 两个异步任务的合并:泡茶喝水 下面的代码中我们实现泡茶喝水。这里分3个任务:任务1负责洗水壶、烧开水,任…...
Nginx知识
nginx 精简的配置文件 worker_processes 1; # 可以理解为一个内核一个worker # 开多了可能性能不好events {worker_connections 1024; } # 一个 worker 可以创建的连接数 # 1024 代表默认一般不用改http {include mime.types;# 代表引入的配置文件# mime.types 在 ngi…...
Unity开发游戏使用XLua的基础
Unity使用Xlua的常用编码方式,做一下记录 1、C#调用lua 1、Lua解析器 private LuaEnv env new LuaEnv();//保持它的唯一性void Start(){env.DoString("print(你好lua)");//env.DoString("require(Main)"); 默认在resources文件夹下面//帮助…...

AI-ISP论文Learning to See in the Dark解读
论文地址:Learning to See in the Dark 图1. 利用卷积网络进行极微光成像。黑暗的室内环境。相机处的照度小于0.1勒克斯。索尼α7S II传感器曝光时间为1/30秒。(a) 相机在ISO 8000下拍摄的图像。(b) 相机在ISO 409600下拍摄的图像。该图像存在噪点和色彩偏差。©…...

OpenCV:开运算
目录 1. 简述 2. 用腐蚀和膨胀实现开运算 2.1 代码示例 2.2 运行结果 3. 开运算接口 3.1 参数详解 3.2 代码示例 3.3 运行结果 4. 开运算应用场景 5. 注意事项 6. 总结 相关阅读 OpenCV:图像的腐蚀与膨胀-CSDN博客 OpenCV:闭运算-CSDN博客 …...
38. RTC实验
一、RTC原理详解 1、6U内部自带到了一个RTC外设,确切的说是SRTC。6U和6ULL的RTC内容在SNVS章节。6U的RTC分为LP和HP。LP叫做SRTC,HP是RTC,但是HP的RTC掉电以后数据就丢失了,即使用了纽扣电池也没用。所以必须要使用LP,…...
Flutter 新春第一弹,Dart 宏功能推进暂停,后续专注定制数据处理支持
在去年春节,Flutter 官方发布了宏(Macros)编程的原型支持, 同年的 5 月份在 Google I/O 发布的 Dart 3.4 宣布了宏的实验性支持,但是对于 Dart 内部来说,从启动宏编程实验开始已经过去了几年,但…...
巴菲特价值投资思想的核心原则
巴菲特价值投资思想的核心原则 关键词:安全边际、长期投资、内在价值、管理团队、经济护城河、简单透明 摘要:本文深入探讨了巴菲特价值投资思想的核心原则,包括安全边际、长期投资、企业内在价值、优秀管理团队、经济护城河和简单透明的业务…...
C 或 C++ 中用于表示常量的后缀:1ULL
1ULL 是一个在 C 或 C 中用于表示常量的后缀,它具体指示编译器将这个数值视为特定类型的整数。让我们详细解释一下: 1ULL 的含义 1: 这是最基本的部分,表示数值 1。U: 表示该数值是无符号(Unsigned)的。这意味着它只…...
vue3中el-input无法获得焦点的问题
文章目录 现象两次nextTick()加setTimeout()解决结论 现象 el-input被外层div包裹了,设置autofocus不起作用: <el-dialog v-model"visible" :title"title" :append-to-bodytrue width"50%"><el-form v-model&q…...

程序诗篇里的灵动笔触:指针绘就数据的梦幻蓝图<3>
大家好啊,我是小象٩(๑ω๑)۶ 我的博客:Xiao Xiangζั͡ޓއއ 很高兴见到大家,希望能够和大家一起交流学习,共同进步。 今天我们来对上一节做一些小补充,了解学习一下assert断言,指针的使用和传址调用…...

(三)QT——信号与槽机制——计数器程序
目录 前言 信号(Signal)与槽(Slot)的定义 一、系统自带的信号和槽 二、自定义信号和槽 三、信号和槽的扩展 四、Lambda 表达式 总结 前言 信号与槽机制是 Qt 中的一种重要的通信机制,用于不同对象之间的事件响…...
mongodb源码分析session执行handleRequest命令find过程
mongo/transport/service_state_machine.cpp已经分析startSession创建ASIOSession过程,并且验证connection是否超过限制ASIOSession和connection是循环接受客户端命令,把数据流转换成Message,状态转变流程是:State::Created 》 St…...

Opencv中的addweighted函数
一.addweighted函数作用 addweighted()是OpenCV库中用于图像处理的函数,主要功能是将两个输入图像(尺寸和类型相同)按照指定的权重进行加权叠加(图像融合),并添加一个标量值&#x…...

家政维修平台实战20:权限设计
目录 1 获取工人信息2 搭建工人入口3 权限判断总结 目前我们已经搭建好了基础的用户体系,主要是分成几个表,用户表我们是记录用户的基础信息,包括手机、昵称、头像。而工人和员工各有各的表。那么就有一个问题,不同的角色…...

Ascend NPU上适配Step-Audio模型
1 概述 1.1 简述 Step-Audio 是业界首个集语音理解与生成控制一体化的产品级开源实时语音对话系统,支持多语言对话(如 中文,英文,日语),语音情感(如 开心,悲伤)&#x…...

【Java_EE】Spring MVC
目录 Spring Web MVC 编辑注解 RestController RequestMapping RequestParam RequestParam RequestBody PathVariable RequestPart 参数传递 注意事项 编辑参数重命名 RequestParam 编辑编辑传递集合 RequestParam 传递JSON数据 编辑RequestBody …...

第 86 场周赛:矩阵中的幻方、钥匙和房间、将数组拆分成斐波那契序列、猜猜这个单词
Q1、[中等] 矩阵中的幻方 1、题目描述 3 x 3 的幻方是一个填充有 从 1 到 9 的不同数字的 3 x 3 矩阵,其中每行,每列以及两条对角线上的各数之和都相等。 给定一个由整数组成的row x col 的 grid,其中有多少个 3 3 的 “幻方” 子矩阵&am…...

全志A40i android7.1 调试信息打印串口由uart0改为uart3
一,概述 1. 目的 将调试信息打印串口由uart0改为uart3。 2. 版本信息 Uboot版本:2014.07; Kernel版本:Linux-3.10; 二,Uboot 1. sys_config.fex改动 使能uart3(TX:PH00 RX:PH01),并让boo…...

Map相关知识
数据结构 二叉树 二叉树,顾名思义,每个节点最多有两个“叉”,也就是两个子节点,分别是左子 节点和右子节点。不过,二叉树并不要求每个节点都有两个子节点,有的节点只 有左子节点,有的节点只有…...

Reasoning over Uncertain Text by Generative Large Language Models
https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829 1. 概述 文本中的不确定性在许多语境中传达,从日常对话到特定领域的文档(例如医学文档)(Heritage 2013;Landmark、Gulbrandsen 和 Svenevei…...

JVM 内存结构 详解
内存结构 运行时数据区: Java虚拟机在运行Java程序过程中管理的内存区域。 程序计数器: 线程私有,程序控制流的指示器,分支、循环、跳转、异常处理、线程恢复等基础功能都依赖这个计数器完成。 每个线程都有一个程序计数…...