JVM常见面试题汇总(强推!!!)

JVM

JVM组成

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

在JVM中共有四大部分,分别是ClassLoader(类加载器)、RuntimeDataArea(运行时数据区,内存分区)、ExecutionEngine(执行引擎)、NativeMethod Library(本地库接口)

它们的运行流程是:

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

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

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

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

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

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

3.详细介绍一下程序计数器的作用

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

4.详细介绍Java堆

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

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

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

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

5.解释一下方法区

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

6.什么是直接内存?

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

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

7.什么是虚拟机栈?

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

8.堆栈的区别

  • 栈内存一般会用来存储局部变量和方法调用,但堆内存是用来存储Java对象和数组的。堆会GC垃圾回收,而栈不会。
  • 第二、栈内存是线程私有的,而堆内存是线程共有的。
  • 第三、两者异常错误不同,但如果栈内存或者堆内存不足都会抛出异常。

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

堆空间不足:java.lang.0utOfMemoryError。

类加载器

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

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

常见的类加载器有4个:

  1. 第一个是启动类加载器(BootStrap ClassLoader):其是由C++编写实现。用于加载JAVA HOME/ire/lib目录下的类库,
  2. 第二个是扩展类加载器(ExtClassLoader):该类是ClassLoader的子类,主要加载JAVA HOME/ire/lib/ext目录中的类库
  3. 第三个是应用类加载器(AppClassLoader):该类是ClassLoader的子类,主要用于加载classPath下的类,也就是加载开发者自己编写的Java类
  4. 第四个是自定义类加载器:开发者自定义类继承ClassLoader,实现自定义类加载规则。

2.说一下类装载执行过程

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

  • 加载:查找和导入class文件
  • 验证:保证加载类的准确性
  • 准备:为类变量分配内存并设置类变量初始值
  • 解析:把类中的符号引用转换为直接引用
  • 初始化:对类的静态变量,静态代码块执行初始化操作
  • 使用:JVM开始从入口方法开始执行用户的程序代码
  • 卸载:当用户程序代码执行完毕后,JVM便开始销毁创建的Class 对象,最后负责运行的 JVM 也退出内存

3.什么是双亲委派模型?

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

垃圾回收

1.简述Java垃圾回收机制(GC是什么?为什么要GC)

为了让程序员更专注于代码的实现,而不用过多的考虑内存释放的问题,所以,在Java语言中,有了自动的垃圾回收机制,也就是我们熟悉的GC(Garbage Collection)。有了垃圾回收机制后,程序员只需要关心内存的申请即可,内存的释放由系统自动识别完成。在进行垃圾回收时,不同的对象引用类型,GC会采用不同的回收时机。

2. 强引用、软引用、弱引用、虚引用的区别?

  • 强引用最为普通的引用方式,表示一个对象处于有用且必须的状态,如果一个对象具有强引用,则GC并不会回收它。即便堆中内存不足了,宁可出现OOM,也不会对其进行回收
  • 软引用表示一个对象处于有用且非必须状态,如果一个对象处于软引用,在内存空间足够的情况下,GC机制并不会回收它,而在内存空间不足时,则会在00M异常出现之间对其进行回收。但值得注意的是,因为GC线程优先级较低,软引用并不会立即被回收。
  • 弱引用表示一个对象处于可能有用且非必须的状态。在GC线程扫描内存区域时,一旦发现弱引用,就会回收到弱引用相关联的对象。对于弱引用的回收,无关内存区域是否足够,一旦发现则会被回收。同样的,因为GC线程优先级较低,所以弱引用也并不是会被立刻回收。
  • 虚引用表示一个对象处于无用的状态。在任何时候都有可能被垃圾回收。虚引用的使用必须和引用队列Reference Queue联合使用

3.对象什么时候可以被垃圾回收?

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

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

       引用计数法:为每个对象维护一个引用计数器,表示有多少个其他对象引用了该对象。当一个新的引用指向该对象时,引用计数器加1。当一个引用被删除时,引用计数器减1。当引用计数器变为零时,该对象被认为是不再使用的,可以被垃圾回收。缺点是无法处理循环依赖问题。

       可达性分析算法:通过分析程序中根对象(GC root objects)与其他对象之间的引用关系来确定哪些对象是可达的(reachable)。根对象通常包括全局变量、栈变量和寄存器中的变量。从根对象开始,遍历所有可以直接或间接访问的对象,这些对象被标记为可达。未被标记为可达的对象被认为是不可达的,可以被垃圾回收。可以解决循环引用,但是实现较为复杂。

4.JVM垃圾回收算法有哪些?

JVM垃圾回收算法一共有四种:分别是标记清除算法、复制算法、标记整理算法、分代回收。

  • 标记清除算法:垃圾回收分为2个阶段,分别是标记和清除,效率高,有磁盘碎片,内存不连续。
  • 标记整理算法:标记清除算法一样,将存活对象都向内存另一端移动,然后清理边界以外的垃圾,无碎片,对象需要移动,效率低。
  • 复制算法:将原有的内存空间一分为二,每次只用其中的一块,正在使用的对象复制到另一个内存空间中,然后将该内存空间清空,交换两个内存的角色,完成垃圾的回收;无碎片,内存使用率低。

5.详细介绍分代回收

在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的时候,会直接存放到老年代。当老年代满了之后,触发FuIIGC。FullGC同时回收新生代和老年代,当前只会存在一个FullGC的线程进行执行,其他的线程全部会被挂起。我们在程序中要尽量避免FullGC的出现。

6.说一下新生代、老年代、永久代的区别

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

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

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

7.JVM有哪些垃圾回收器?

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

1)串行垃圾收集器:Serial和Serial Old串行垃圾收集器,是指使用单线程进行垃圾回收,堆内存较小,适合个人电脑。Serial 作用于新生代,采用复制算法。Serial Old 作用于老年代,采用标记-整理算法。垃圾回收时,只有一个线程在工作,并且java应用中的所有线程都要暂停(STW),等待垃圾回收的完成。
(2)并行垃圾收集器:Parallel New和Parallel Old是一个并行垃圾回收器,JDK8默认使用此垃圾回收器。Parallel New作用于新生代,采用复制算法。Parallel Old作用于老年代,采用标记-整理算法。垃圾回收时,多个线程在工作,并且java应用中的所有线程都要暂停(STW),等待垃圾回收的完成。

(3)CMS(并发)垃圾收集器:CMS全称 Concurrent Mark Sweep,是一款并发的、使用标记-清除算法的垃圾回收器,该回收器是针对老年代垃圾回收的,是一款以获取最短回收停顿时间为目标的收集器,停顿时间短,用户体验就好。其最大特点是在进行垃圾回收时,应用仍然能正常运行。

8.详细介绍G1垃圾收集器

G1(Garbage First)垃圾回收器是Oracle在JDK 7中引入的面向多核机器和大内存应用的垃圾回收器。目的在于替代原有的CMS(Concurrent Mark-Sweep)垃圾回收器,并提供更加一致的停顿时间以及更好的性能。G1垃圾回收器通过划分堆内存为多个大小相等的独立区域(regions),并优先回收垃圾最多的区域(Garbage First),实现了更加灵活和高效的内存管理。

主要特性:

  • 区域划分:堆内存被划分为多个大小相等的区域,每个区域可以在不同时间充当Eden、Survivor或Old区的一部分。
  • 暂停时间预测:G1垃圾回收器可以根据设定的暂停时间目标,动态调整每次回收的工作量,尽量保证应用的响应时间。
  • 并发和并行:G1在垃圾回收过程中大量使用并发和并行技术,充分利用多核CPU资源,减少停顿时间。
  • 混合回收:G1不仅回收年轻代(Young Generation)的对象,也会定期回收老年代(Old Generation)中的对象,避免了CMS的碎片化问题。
  • 整合的内存整理:G1在回收过程中会进行内存整理,减少内存碎片,提高内存利用效率。

工作机制:

G1垃圾回收器的工作机制主要包括以下几个阶段:

  • 初始标记(Initial Marking):标记所有从根对象(GC roots)直接可达的对象。这阶段会引发一次短暂的停顿(STW, Stop The World)。
  • 并发标记(Concurrent Marking):从GC roots开始,对堆中的对象进行并发标记,找出存活对象。这阶段大部分是在应用运行时进行的,不会停止应用线程。
  • 最终标记(Final Marking):完成标记周期,处理在并发标记阶段被修改的对象(通过Remembered Set记录)。这阶段会引发一次短暂的停顿。
  • 筛选回收(Live Data Counting and Cleanup):根据标记结果,识别并选择需要回收的区域(regions)。主要回收那些包含最多垃圾对象的区域,优先回收垃圾最多的区域。进行回收和内存整理,回收阶段会引发一次STW停顿。

9. Minor GC、Major GC、Full GC是什么

它们指的是不同代之间的垃圾回收;

  • Minor GC 发生在新生代的垃圾回收,暂停时间短
  • Major GC 老年代区域的垃圾回收,老年代空间不足时,会先尝试触发MinorGC。Minor GC之后空间还不足,则会触发MajorGC,Major GC速度比较慢,暂停时间长。
  • Full GC新生代+老年代完整垃圾回收,暂停时间长,应尽力避免。

JVM调优

1. JVM 调优的参数有哪些?

我记得当时我们设置过堆的大小,像-Xms和-Xmx,还有就是可以设置年轻代中Eden区和两个Survivor区的大小比例,还有就是可以设置使用哪种垃圾回收器等等。具体的指令还真记不太清楚。

2. 调试 JVM都有哪些工具?

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

  • jps输出JVM中运行的进程状态信息
  • jstack查看java进程内线程的堆栈信息。
  • jmap 用于生成堆转存快照
  • jstat用于JVM统计监测工具
  • 还有一些可视化工具,像jconsole和VisualVM等

3.java内存泄漏排查思路

  • 第一,通过jmap指定打印他的内存快照 dump文件,不过有的情况打印不了,我们会设置vm参数让程序自动生成dump文件。
  • 第二,通过工具去分析 dump文件,jdk自带的VisualVM就可以分析
  • 第三,通过查看堆信息的情况,可以大概定位内存溢出是哪行代码出了问题
  • 第四,找到对应的代码,通过阅读上下文的情况,进行修复即可

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

  • 第一可以使用使用top命令查看占用cpu的情况
  • 第二通过top命令查看后,可以查看是哪一个进程占用cpu较高,记录这个进程id
  • 第三可以通过ps 查看当前进程中的线程信息,看看哪个线程的cpu占用较高
  • 第四可以jstack命令打印进行的id,找到这个线程,就可以进一步定位问题代码的行号
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值