当前位置: 首页 > news >正文

JVM相关面试八股

什么是双亲委派模型?

如果一个类加载器在接到加载类的请求时,它首先不会自己尝试去加载这个类,而是把这个请求任务委托给父类加载器去完成,依次递归,如果父类加载器可以完成类加载任务,就返回成功;只有父类加载器无法完成此加载任务时,才由下一级去加载

JVM为什么采用双亲委派机制
  • 通过双亲委派机制可以避免某一个类被重复加载,当父类已经加载后则无需重复加载,保证唯一性

  • 为了安全,保证类库 API 不会被修改

JVM由哪些部分组成,运行流程是什么?

JVM 中有四大部分,分别是类加载器、运行时数据区,内存分区、执行引擎、本地库接口

运行流程是:

第一,类加载器把 Java 代码转换为字节码

第二,运行时数据区把字节码加载到内存中

第三,执行引擎将字节码翻译为底层系统指令,交给CPU执行

介绍一下程序计数器的作用?

记录当前线程正在执行的字节码指令的地址(行号)

说一下堆栈的区别?

第一,栈内存一般会用来存储局部变量和方法调用,但堆内存是用来存储Java对象和数组的的。堆会GC垃圾回收,而栈不会

第二、栈内存是线程私有的,而堆内存是线程共有的

第三、两者异常错误不同,但如果栈内存或者堆内存不足都会抛出异常

栈空间不足:java.lang.StackOverFlowError

堆空间不足:java.lang.OutOfMemoryError

什么是类加载器,类加载器有哪些?

类加载器(ClassLoader)的作用就是将字节码文件加载到 JVM 中,从而启动Java 程序

常见的类加载器:

  • 启动类加载器(BootStrap ClassLoader):用于加载 JAVA_HOME/jre/lib 目录下的类库

  • 扩展类加载器(ExtClassLoader):加载 JAVA_HOME/jre/lib/ext目录中的类库

  • 应用类加载器(AppClassLoader):用于加载 classPath 下的类,也就是开发者自己编写的Java

  • 自定义类加载器:开发者自定义类继承 ClassLoader,实现自定义类加载规则

说一下类装载的执行过程?

整个生命周期包括:加载、验证、准备、解析、初始化、使用和卸载7个阶段。其中,验证、准备和解析这三个部分统称为连接

1.加载:查找和导入 class 文件

2.验证:保证加载类的准确性

3.准备:为类变量分配内存并设置类变量初始值

4.解析:把类中的符号引用转换为直接引用

5.初始化:初始化类的静态变量,静态代码块

6.使用:JVM 从入口方法开始执行用户的程序代码

7.卸载:当用户程序代码执行完后,JVM 开始销毁创建的 Class 对象,最后 JVM 也退出内存

简述Java垃圾回收机制?(GC是什么?为什么要GC)
  • 为了让程序员更专注于代码的实现,而不用考虑内存释放的问题,所以有了自动的垃圾回收机制,也就是GC

  • 有了垃圾回收机制后,程序员只需要关心内存的申请即可,不用关心内存的释放

强引用、软引用、弱引用、虚引用的区别?
  • 强引用表示一个对象处于有用且必须的状态,如果一个对象具有强引用,则 GC 不会回收它。即便堆中内存不足,出现 OOM (OutOfMemoryError,内存资源耗尽),也不会对其进行回收

  • 软引用表示一个对象处于有用且非必须状态,在内存充足时保留对象,而在内存不足时允许垃圾回收器(GC)自动回收这些对象​​

  • 弱引用表示一个对象处于可能有用且非必须的状态。无论内存是否充足,​​GC运行时发现该对象都会将其回收​​

  • 虚引用表示一个对象处于无用的状态。在任何时候都有可能被垃圾回收

对象什么时候可以被垃圾器回收

如果一个或多个对象没有任何的引用指向它了,那么这个对象有可能会被回收

如果要定位什么是垃圾,有两种方式,一是引用计数法,二是可达性分析算法

通常使用可达性分析算法来确定是不是垃圾

说一下 JVM 有哪些垃圾回收器?

jvm 中有多种垃圾收集器,包括:串行垃圾收集器、并行垃圾收集器(JDK8默认)、CMS(并发)垃圾收集器、G1垃圾收集器(JDK9默认)

JVM 垃圾回收算法有哪些?

一共有四种,分别是标记清除算法、复制算法、标记整理算法、分代回收

假如项目中产生了java内存泄露,说一下你的排查思路?

首先可以通过 jmap 指定打印他的内存快照 dump 文件,不过有的时候打印不了,可以设置 jvm参数让程序自动生成 dump 文件

然后通过工具去分析 dump 文件,jdk 自带的 VisualVM 就可以分析

然后查看堆信息的情况,可以大概定位内存溢出是哪行代码出了问题

找到对应的代码,通过阅读上下文的情况,进行修复

服务器CPU持续飙高,你的排查方案与思路?

第一可以使用使用 top 命令查看占用 cpu 的情况

然后查看是哪一个进程占用 cpu 较高,记录这个进程 id

第三通过 ps 查看当前进程中的线程信息,看看哪个线程的 cpu 占用较高

再通过 jstack 命令打印进行的 id,找到这个线程,就可以进一步定位问题代码的行号

JVM 调优的参数可以在哪里设置参数值?

springboot项目在项目启动的时候,java -jar 中加入参数就行了

面试官:用的 JVM 调优的参数都有哪些?

候选人:

嗯,这些参数是比较多的

我记得当时我们设置过堆的大小,像-Xms和-Xmx

还有就是可以设置年轻代中Eden区和两个Survivor区的大小比例

还有就是可以设置使用哪种垃圾回收器等等。具体的指令还真记不太清楚。

调试 JVM都用了哪些工具?

jdk 自带的一些工具,比如:

jps 输出JVM中运行的进程状态信息

jstack 查看 java 进程内线程的堆栈信息

jmap 用于生成堆转存快照

jstat 用于 JVM 统计监测工具

还有一些可视化工具,像 jconsoleVisualVM

详细说一下 JVM 运行时数据区?

运行时数据区包含堆、方法区、栈、本地方法栈、程序计数器

  • 堆解决的是对象实例存储的问题

  • 方法区可以认为是堆的一部分,用于存储已被虚拟机加载的信息,常量、静态变量、即时编译器编译后的代码

  • 栈解决的是程序运行的问题,栈里面存的是栈帧,栈帧里面存的是局部变量表、操作数栈、动态链接、方法出口等信息

  • 本地方法栈与栈功能相同,本地方法栈执行的是本地方法,是Java 调用非 Java 代码的接口

  • 程序计数器(PC寄存器)程序计数器中存放的是当前线程所执行的字节码的行数。JVM 工作时就是通过改变这个计数器的值来选取下一个需要执行的字节码指令

你能详细聊一下分代回收吗?

关于分代回收是这样的

在java8时,堆被分为了两份:新生代和老年代,它们默认空间占用比例是1:2

对于新生代,内部又被分为了三个区域。Eden区,S0区,S1区默认空间占用比例是8:1:1

具体的工作机制是:

1)当创建一个对象的时候,那么这个对象会被分配在新生代的Eden区。当Eden区要满了时候,触发YoungGC。

2)当进行YoungGC后,此时在Eden区存活的对象被移动到S0区,并且当前对象的年龄会加1,清空Eden区。

3)当再一次触发YoungGC的时候,会把Eden区中存活下来的对象和S0中的对象,移动到S1区中,这些对象的年龄会加1,清空Eden区和S0区。

4)当再一次触发YoungGC的时候,会把Eden区中存活下来的对象和S1中的对象,移动到S0区中,这些对象的年龄会加1,清空Eden区和S1区。

5)对象的年龄达到了某一个限定的值(默认15岁 ),那么这个对象就会进入到老年代中。

当然也有特殊情况,如果进入Eden区的是一个大对象,在触发YoungGC的时候,会直接存放到老年代

当老年代满了之后,触发FullGCFullGC同时回收新生代和老年代,当前只会存在一个FullGC的线程进行执行,其他的线程全部会被挂起。 我们在程序中要尽量避免FullGC的出现。

讲一下新生代、老年代、永久代的区别?

新生代主要用来存放新生的对象。

老年代主要存放应用中生命周期长的内存对象。

永久代指的是永久保存区域。主要存放Class和Meta(元数据)的信息。在Java8中,永久代已经被移除,取而代之的是一个称之为“元数据区”(元空间)的区域。元空间和永久代类似,不过元空间与永久代之间最大的区别在于:元空间并不在虚拟机中,而是使用本地内存。因此,默认情况下,元空间的大小仅受本地内存的限制。

http://www.dtcms.com/a/298032.html

相关文章:

  • Mysql实现高可用(主从、集群)
  • Unity GC 系列教程第五篇:高级 GC 内核
  • Python(32)Python内置函数全解析:30个核心函数的语法、案例与最佳实践
  • IPv6网络排障详细步骤指南(附工具命令+配置检查点+典型案例)
  • 【MAC的VSCode使用】
  • Python 爬虫实战指南:按关键字搜索淘宝商品
  • 使用Redis实现MySQL的数据缓存
  • JavaScript前端加密技术:aes.js与crypto-js.js深度解析
  • 2025年7月25日训练日志
  • PWM信号控制电机
  • Creo 模块众多,企业如何按需灵活分配许可证资源?
  • 倒计时 1 天!深思考携超小端侧多模态大模型,在2025 WAIC H2-427展位等你解锁端侧新可能!
  • go语言基础教程:【2】基础语法:基本数据类型(整形和浮点型)
  • mybatis 差异更新法
  • gig-gitignore工具实战开发(三):gig add基础实现
  • k8s的service、deployment、探针详解
  • vue2用elementUI做单选下拉树
  • HC32 中断实现
  • ubuntu上将TempMonitor加入开机自动运行的方法
  • Python异常,模块与包
  • 电厂液压执行器自动化升级:Modbus TCP与DeviceNet的协议贯通实践
  • 从热点到刚需:SmartMediaKit为何聚焦B端视频系统建设?
  • 「iOS」——GCD其他方法详解
  • 自然语言处理技术应用领域深度解析:从理论到实践的全面探索
  • Unity 多人游戏框架学习系列十一
  • http-proxy-middleware MaxListenersExceededWarning
  • [2025CVPR-图象分类方向]SPARC:用于视觉语言模型中零样本多标签识别的分数提示和自适应融合
  • 【STM32】FreeRTOS任务的挂起与解挂(四)
  • 学习游戏制作记录(克隆技能)7.25
  • 踩坑记录:因版本不匹配导致 Boost 1.85 编译失败的完整解决过程