【Java开发面试系列】JVM相关面试题(精选)

【Java开发面试系列】JVM相关面试题(精选)

文章目录

  • 【Java开发面试系列】JVM相关面试题(精选)
    • 前言
      • 一、JVM组成
      • 二、类加载器
      • 三、垃圾回收
      • 四、JVM实践(调优)

🌈你好呀!我是 山顶风景独好
💝欢迎来到我的博客,很高兴能够在这里和您见面!
💝希望您在这里可以感受到一份轻松愉快的氛围!
💝不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。
🚀 欢迎一起踏上探险之旅,挖掘无限可能,共同成长!

前言

在寻求面试资料的旅途中,我有幸发现了黑马老师所撰写的宝贵资料,其内容详尽且见解独到,对我而言无疑是一笔巨大的财富。我深感其智慧与经验的深厚,并在此过程中受益匪浅。

为了深入学习和巩固这些宝贵的知识,我精心整理了所有的面试题目,并编辑成这一系列的博客。这不仅是我个人学习的读书笔记,更是我对黑马老师无私奉献和深厚学识的崇高敬意。

同时,我也希望通过分享这些博客,将这份知识与智慧传递给更多的同学,共同提高我们的面试技巧和知识储备

  • 本系列文章定位:
    【Java开发面试系列】MySQL相关面试题(精选)
    【Java开发面试系列】Redis相关面试题(精选)
    【Java开发面试系列】框架相关面试题(精选)
    【Java开发面试系列】微服务相关面试题(精选)
    【Java开发面试系列】消息中间件相关面试题(精选)
    【Java开发面试系列】常见集合相关面试题(精选)
    【Java开发面试系列】并发编程相关面试题(精选)
    【Java开发面试系列】JVM相关面试题(精选)
    【Java开发面试系列】设计模式相关面试题(精选)
    【Java开发面试系列】技术场景相关面试题(精选)

一、JVM组成

面试官:JVM由那些部分组成,运行流程是什么?

候选人:

嗯,好的~~

在JVM中共有四大部分,分别是ClassLoader(类加载器)、Runtime Data Area(运行时数据区,内存分区)、Execution Engine(执行引擎)、Native Method Library(本地库接口)

它们的运行流程是:

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

  • 第二,运行时数据区(Runtime Data Area)把字节码加载到内存中,而字节码文件只是JVM的一套指令集规范,并不能直接交给底层系统去执行,而是有执行引擎运行

  • 第三,执行引擎(Execution Engine)将字节码翻译为底层系统指令,再交由CPU执行去执行,此时需要调用其他语言的本地库接口(Native Method Library)来实现整个程序的功能。


面试官:好的,你能详细说一下 JVM 运行时数据区吗?

候选人:

嗯,好~

运行时数据区包含了堆、方法区、栈、本地方法栈、程序计数器这几部分,每个功能作用不一样。

  • 堆解决的是对象实例存储的问题,垃圾回收器管理的主要区域。
  • 方法区可以认为是堆的一部分,用于存储已被虚拟机加载的信息,常量、静态变量、即时编译器编译后的代码。
  • 栈解决的是程序运行的问题,栈里面存的是栈帧,栈帧里面存的是局部变量表、操作数栈、动态链接、方法出口等信息。
  • 本地方法栈与栈功能相同,本地方法栈执行的是本地方法,一个Java调用非Java代码的接口。
  • 程序计数器(PC寄存器)程序计数器中存放的是当前线程所执行的字节码的行数。JVM工作时就是通过改变这个计数器的值来选取下一个需要执行的字节码指令。

面试官:好的,你再详细介绍一下程序计数器的作用?

候选人:

嗯,是这样~~

java虚拟机对于多线程是通过线程轮流切换并且分配线程执行时间。在任何的一个时间点上,一个处理器只会处理执行一个线程,如果当前被执行的这个线程它所分配的执行时间用完了【挂起】。处理器会切换到另外的一个线程上来进行执行。并且这个线程的执行时间用完了,接着处理器就会又来执行被挂起的这个线程。这时候程序计数器就起到了关键作用,程序计数器在来回切换的线程中记录他上一次执行的行号,然后接着继续向下执行。


面试官:你能给我详细的介绍Java堆吗?

候选人:

  • 好的~

  • Java中的堆术语线程共享的区域。主要用来保存对象实例,数组等,当堆中没有内存空间可分配给实例,也无法再扩展时,则抛出OutOfMemoryError异常。

​在JAVA8中堆内会存在年轻代、老年代

  1. Young区被划分为三部分,Eden区和两个大小严格相同的Survivor区,其中,Survivor区间中,某一时刻只有其中一个是被使用的,另外一个留做垃圾收集时复制对象用。在Eden区变满的时候, GC就会将存活的对象移到空闲的Survivor区间中,根据JVM的策略,在经过几次垃圾收集后,任然存活于Survivor的对象将被移动到Tenured区间。

  2. Tenured区主要保存生命周期长的对象,一般是一些老的对象,当一些对象在Young复制转移一定的次数以后,对象就会被转移到Tenured区。


面试官:能不能解释一下方法区?

候选人:

好的~

与虚拟机栈类似。本地方法栈是为虚拟机执行本地方法时提供服务的。不需要进行GC。本地方法一般是由其他语言编写。


面试官:你听过直接内存吗?

候选人:

嗯~~

  • 它又叫做堆外内存线程共享的区域,在 Java 8 之前有个永久代的概念,实际上指的是 HotSpot 虚拟机上的永久代,它用永久代实现了 JVM 规范定义的方法区功能,主要存储类的信息,常量,静态变量,即时编译器编译后代码等,这部分由于是在堆中实现的,受 GC 的管理,不过由于永久代有 -XX:MaxPermSize 的上限,所以如果大量动态生成类(将类信息放入永久代),很容易造成 OOM,有人说可以把永久代设置得足够大,但很难确定一个合适的大小,受类数量,常量数量的多少影响很大。

  • 所以在 Java 8 中就把方法区的实现移到了本地内存中的元空间中,这样方法区就不受 JVM 的控制了,也就不会进行 GC,也因此提升了性能。


面试官:什么是虚拟机栈

候选人:

虚拟机栈是描述的是方法执行时的内存模型,是线程私有的,生命周期与线程相同,每个方法被执行的同时会创建栈桢。保存执行方法时的局部变量、动态连接信息、方法返回地址信息等等。方法开始执行的时候会进栈,方法执行完会出栈【相当于清空了数据】,所以这块区域不需要进行 GC


面试官:能说一下堆栈的区别是什么吗?

候选人:

嗯,好的,有这几个区别

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

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

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

  • 栈空间不足:java.lang.StackOverFlowError。

  • 堆空间不足:java.lang.OutOfMemoryError。

二、类加载器

面试官:什么是类加载器,类加载器有哪些?

候选人:

嗯,是这样的

JVM只会运行二进制文件,而类加载器(ClassLoader)的主要作用就是将字节码文件加载到JVM中,从而让Java程序能够启动起来。

常见的类加载器有4个

  • 第一个是启动类加载器(BootStrap ClassLoader):其是由C++编写实现。用于加载JAVA_HOME/jre/lib目录下的类库。

  • 第二个是扩展类加载器(ExtClassLoader):该类是ClassLoader的子类,主要加载JAVA_HOME/jre/lib/ext目录中的类库。

  • 第三个是应用类加载器(AppClassLoader):该类是ClassLoader的子类,主要用于加载classPath下的类,也就是加载开发者自己编写的Java类。

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


面试官:说一下类装载的执行过程?

候选人:

嗯,这个过程还是挺多的。

类从加载到虚拟机中开始,直到卸载为止,它的整个生命周期包括了:加载、验证、准备、解析、初始化、使用和卸载这7个阶段。其中,验证、准备和解析这三个部分统称为连接(linking)

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

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

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

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

  5. 初始化:对类的静态变量,静态代码块执行初始化操作

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

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


面试官:什么是双亲委派模型?

候选人:

嗯,它是是这样的。

如果一个类加载器收到了类加载的请求,它首先不会自己尝试加载这个类,而是把这请求委派给父类加载器去完成,每一个层次的类加载器都是如此,因此所有的加载请求最终都应该传说到顶层的启动类加载器中,只有当父类加载器返回自己无法完成这个加载请求(它的搜索返回中没有找到所需的类)时,子类加载器才会尝试自己去加载


面试官:JVM为什么采用双亲委派机制

候选人:

主要有两个原因。

  • 第一、通过双亲委派机制可以避免某一个类被重复加载,当父类已经加载后则无需重复加载,保证唯一性。

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

三、垃圾回收

面试官:简述Java垃圾回收机制?(GC是什么?为什么要GC)

候选人:

  • 嗯,是这样~~

  • 为了让程序员更专注于代码的实现,而不用过多的考虑内存释放的问题,所以,在Java语言中,有了自动的垃圾回收机制,也就是我们熟悉的GC(Garbage Collection)。

  • 有了垃圾回收机制后,程序员只需要关心内存的申请即可,内存的释放由系统自动识别完成。

  • 在进行垃圾回收时,不同的对象引用类型,GC会采用不同的回收时机


面试官:强引用、软引用、弱引用、虚引用的区别?

候选人:

  • 嗯嗯~

  • 强引用最为普通的引用方式,表示一个对象处于有用且必须的状态,如果一个对象具有强引用,则GC并不会回收它。即便堆中内存不足了,宁可出现OOM,也不会对其进行回收

  • 软引用表示一个对象处于有用且非必须状态,如果一个对象处于软引用,在内存空间足够的情况下,GC机制并不会回收它,而在内存空间不足时,则会在OOM异常出现之间对其进行回收。但值得注意的是,因为GC线程优先级较低,软引用并不会立即被回收。

  • 弱引用表示一个对象处于可能有用且非必须的状态。在GC线程扫描内存区域时,一旦发现弱引用,就会回收到弱引用相关联的对象。对于弱引用的回收,无关内存区域是否足够,一旦发现则会被回收。同样的,因为GC线程优先级较低,所以弱引用也并不是会被立刻回收。

  • 虚引用表示一个对象处于无用的状态。在任何时候都有可能被垃圾回收。虚引用的使用必须和引用队列Reference Queue联合使用


面试官:对象什么时候可以被垃圾器回收

候选人:

  • 如果一个或多个对象没有任何的引用指向它了,那么这个对象现在就是垃圾,如果定位了垃圾,则有可能会被垃圾回收器回收。

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

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


面试官: 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中,永久代已经被移除,取而代之的是一个称之为“元数据区”(元空间)的区域。元空间和永久代类似,不过元空间与永久代之间最大的区别在于:元空间并不在虚拟机中,而是使用本地内存。因此,默认情况下,元空间的大小仅受本地内存的限制。


面试官:说一下 JVM 有哪些垃圾回收器?

候选人:

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


面试官:Minor GC、Major GC、Full GC是什么

候选人:

嗯,其实它们指的是不同代之间的垃圾回收

  • Minor GC 发生在新生代的垃圾回收,暂停时间短

  • Major GC 老年代区域的垃圾回收,老年代空间不足时,会先尝试触发Minor GC。Minor GC之后空间还不足,则会触发Major GC,Major GC速度比较慢,暂停时间长

  • Full GC 新生代 + 老年代完整垃圾回收,暂停时间长,应尽力避免

四、JVM实践(调优)

面试官:JVM 调优的参数可以在哪里设置参数值?

候选人:

我们当时的项目是springboot项目,可以在项目启动的时候,java -jar中加入参数就行了


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

候选人:

  • 嗯,这些参数是比较多的

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

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

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


面试官:嗯,好的,你们平时调试 JVM都用了哪些工具呢?

候选人:

嗯,我们一般都是使用jdk自带的一些工具,比如

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

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

  • jmap 用于生成堆转存快照

  • jstat用于JVM统计监测工具

还有一些可视化工具,像jconsole和VisualVM等


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

候选人:

嗯,这个我在之前项目排查过

  • 第一呢可以通过jmap指定打印他的内存快照 dump文件,不过有的情况打印不了,我们会设置vm参数让程序自动生成dump文件

  • 第二,可以通过工具去分析 dump文件,jdk自带的VisualVM就可以分析

  • 第三,通过查看堆信息的情况,可以大概定位内存溢出是哪行代码出了问题

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


面试官:好的,那现在再来说一种情况,就是说服务器CPU持续飙高,你的排查方案与思路?

候选人:

嗯,我思考一下~~

可以这么做~~

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

  • 第二通过top命令查看后,可以查看是哪一个进程占用cpu较高,记录这个进程id

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

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/12484.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【十大排序算法】----C语言版插入排序(详细图解)

目录 一:插入排序——原理 二:插入排序——分析 三:插入排序——实现 四:插入排序——效率 一:插入排序——原理 插入排序的原理和基本思想:把待排序的记录按其关键码值的大小逐个插入到一个已经排好序…

无需重启 NGINX 开源版即可实现 SSL/TLS 证书轮换

原文作者:Maxim Ivanitskiy of F5 原文链接:无需重启 NGINX 开源版即可实现 SSL/TLS 证书轮换 转载来源:NGINX 开源社区 NGINX 唯一中文官方社区 ,尽在 nginx.org.cn 在高性能 Web 服务器领域,NGINX 是一个广受欢迎的…

Django使用

一、根目录下安装 pip install django 二、创建djiango项目 django-admin startproject 项目名称 三、创建app python manage.py startapp app名称 四、启动 python manage.py runserver 五、编写URL与视图关系,相对路径 1、manage.py(见资源绑定…

mysql 8 创建用户并赋予改用户指定数据库权限

一、使用客户端工具登录mysql 二、创建用户 -- 低版本数据库 create user 用户名% identified by 密码; -- 高版本数据库 create user 用户名% identified with mysql_native_password by 密码; -- 示例1: create user test% identified with mysql_native_passwor…

apt和apt-get有什么区别

2024年5月15日,周三上午 apt 和 apt-get 都是 Debian 及其衍生版(如 Ubuntu)中用于软件包管理的工具,但它们之间存在一些差异。 功能和目的: apt 是 apt-get 的改进版本,提供了更简洁和更直观的命令选项。…

多元化、高辨识显示丨基于G32A1445的汽车尾灯解决方案

由刹车灯、倒车灯、转向灯、雾灯等组成的汽车尾灯,既能在光线低暗时发出照明信息,也可向周围环境传递车辆的行驶状态与意图信号,对于行车安全起着至关重要的作用。与传统尾灯相比,贯穿式汽车尾灯更加醒目、美观、安全,…

CSS2(一):CSS选择器

文章目录 1、CSS基础1.1 CSS简介1.2 CSS编写位置1.2.1 行内样式1.2.2 内部样式1.2.3 外部样式1.2.4 样式优先级 1.2.5 CSS代码风格 2、CSS选择器2.1、基本选择器2.1.1 通配选择器2.1.2 元素选择器2.1.3 类选择器2.1.4 ID选择器2.1.5 总结 2.2、CSS复合选择器2.2.1 交集选择器2.…

海外媒体宣发:新加坡.马来西亚如何在海外媒体投放新闻通稿-大舍传媒

导言 随着全球化的进程加速,海外市场对于企业的发展越来越重要。而在海外媒体上宣传企业的新闻通稿,成为了拓展海外市场和提升企业知名度的重要手段之一。本文将介绍大舍传媒对于如何在海外媒体上投放新闻通稿的经验和策略。 准备工作:了解…

Hive 特殊的数据类型 Array、Map、Struct

Array 数组类型,存储数据类型一致的列表数据。 我们可以使用 array 方法来创建一个数组,如下所示: select array(1,2,3,4,5);如果其中的数据类型不一致,那么它会转换成统一的数据类型(前提是能够进行转换&#xff0…

力扣HOT100 - 322. 零钱兑换

解题思路&#xff1a; 动态规划 class Solution {public int coinChange(int[] coins, int amount) {int[] dp new int[amount 1];Arrays.fill(dp, amount 1);dp[0] 0;for (int i 1; i < amount; i) {for (int j 0; j < coins.length; j) {if (coins[j] < i) …

word内容wxml转化html标签对照表

1. 标签 w:document document 文档w:bodybody文档的主体w:sectPrsection——w:pp/div段落w:rspan行内元素w:ttext文本w:tbltable表格w:trtr表格行w:tctd单元格w:brbr换行w:hyperlinka超链接w:roundrectdiv/canvas块w:pictimg图片w:inlinespan行元素w:oMathmath公式w:subsub下标…

宠物管理系统带万字文档

文章目录 宠物管理系统一、项目演示二、项目介绍三、19000字论文参考四、部分功能截图五、部分代码展示六、底部获取项目源码和万字论文参考&#xff08;9.9&#xffe5;带走&#xff09; 宠物管理系统 一、项目演示 宠物管理系统 二、项目介绍 基于springbootvue的前后端分离…

如何让Linux系统崩溃?

如何使 Linux 系统崩溃 警告 下面的代码行是 Bash shell 的一个简短而甜蜜的 fork 炸弹。分叉炸弹之所以有效&#xff0c;是因为它能够产生无限数量的进程。最终&#xff0c;Linux无法处理所有这些&#xff0c;并且会崩溃。 fork 炸弹的一大优点是你不需要 root 权限即可执行它…

Vu2之使用provide与inject调用方法案例

Vu2之使用provide与inject调用方法案例 文章目录 Vu2之使用provide与inject调用方法案例1. 祖先组件使用provide提供方法2. 后代组件使用inject注入并调用方法 在Vue 2中&#xff0c;provide和inject是用于在组件之间传递数据的一种高级技术。虽然它们通常用于传递数据&#xf…

【scikit-learn001】逻辑回归(Logistic Regression)ML模型实战及经验总结(更新中)

1.一直以来想写下基于scikit-learn训练AI算法的系列文章&#xff0c;作为较火的机器学习框架&#xff0c;也是日常项目开发中常用的一款工具&#xff0c;最近刚好挤时间梳理、总结下这块儿的知识体系。 2.熟悉、梳理、总结下scikit-learn框架逻辑回归&#xff08;Logistic Regr…

新串口通道打通纪实

在计算机系统中&#xff0c;串口是“古老”的通信方式&#xff0c;和它同时代的“并口”通信方式已经消失了。但它仍然顽强的存活着&#xff0c;主要原因是在开发和调试底层软件时还经常用到串口。 正因为有这样的需求&#xff0c;幽兰代码本是支持串口的&#xff0c;而且有两种…

vue中父子组件如何相互调用方法

Vue 中父子组件如何相互调用方法 在 Vue 中&#xff0c;父子组件可以通过以下方法相互调用方法&#xff1a; 父组件调用子组件方法 通过 props: 父组件向子组件传递一个 prop&#xff0c;该 prop 是一个函数&#xff0c;子组件可以调用它来触发父组件的方法。通过 refs: 父组…

【现代C++】概念的使用

现代C&#xff08;特别是C20及以后的版本&#xff09;引入了概念&#xff08;Concepts&#xff09;&#xff0c;这是一种指定模板参数必须满足的约束的方式。概念使得模板代码更清晰&#xff0c;更容易理解和使用&#xff0c;并且能在编译时提供更好的错误信息。以下是C概念的关…

UStaticMesh几何数据相关(UE5.2)

UStaticMesh相关类图 UStaticMesh的数据构成 UStaticMesh的FStaticMeshSourceModel UStaticMesh的Mesh几何元数据来自于FStaticMeshSourceModel&#xff0c; 一级Lod就存在一个FStaticMeshSourceModel. FStaticMeshSourceModel几何数据大致包含以下几类: Vertex(点), VertexI…