JVM会有许多零碎但是却很高频的基础考题。牢记这些,才能保证不在面试中落后于人。
说说对象分配规则
这也是之前面试腾讯时候被问到的问题:请介绍JVM如何分配对象?
- 对象优先分配在Eden 区,如果Eden 区没有足够的空间时,虚拟机执行一次Minor GC。
- 大对象直接进入老年代(大对象是指需要大量连续内存空间的对象)。这样做的目的是避
免在Eden 区和两个Survivor 区之间发生大量的内存拷贝(新生代采用复制算法收集内存)。 - 长期存活的对象进入老年代。虚拟机为每个对象定义了一个年龄计数器,如果对象经过了1
次Minor GC 那么对象会进入Survivor 区,之后每经过一次Minor GC 那么对象的年龄加1,
直到达到阈值对象进入老年区。 - 动态判断对象的年龄。如果Survivor 区中相同年龄的所有对象大小的总和大于Survivor
空间的一半,年龄大于或等于该年龄的对象可以直接进入老年代。 - 空间分配担保。每次进行Minor GC 时,JVM 会计算Survivor 区移至老年区的对象的平均
大小, 如果这个值大于老年区的剩余值大小则进行一次Full GC,如果小于检查
HandlePromotionFailure 设置,如果true 则只进行Monitor GC,如果false 则进行Full GC。
说说堆和栈的区别
功能不同
- 栈是运行时单位,代表着逻辑,用于存储方法调用的调用栈和局部变量,所在区域连续,没有碎片;
- 堆是存储单位,代表着数据,可被多个栈共享(包括成员中基本数据类型、引用和引用对象),所在区域不连续,会有碎片。
共享性不同
栈内存是线程私有的。堆内存是所有线程共有的。
异常错误不同
如果栈内存或者堆内存不足都会抛出异常。
- 栈空间不足:java.lang.StackOverFlowError。
- 堆空间不足:java.lang.OutOfMemoryError。
空间大小
栈的空间大小远远小于堆的。在32位的JVM上,通常默认的栈大小是1MB。而在64位的JVM上,默认的栈大小可能会更大一些,通常是2MB。而在32位系统上,默认的堆大小可能在1GB以下,而在64位系统上可能会在2GB到4GB之间。但是这些值只是一个大致的估计,具体的默认堆大小还是需要查看具体的JVM实现和版本。
JVM的永久代中会发生垃圾回收吗?
永久代不会进行垃圾回收,永久代满了,会触发full gc。众所周知永久代(在JDK 8之前)通常用于存储类的元数据、常量池等信息,而不是存储普通对象,所以永久代的垃圾回收由"永久代垃圾收集器"(如CMS)负责,垃圾回收的主要目标是回收不再使用的类元数据和常量,而不是普通的Java对象。
然而,JDK 8引入了元数据区(Metaspace),用于存储类的元数据,取代了永久代。在元数据区中,垃圾回收由不同的机制处理,通常由G1垃圾收集器或者CMS垃圾收集器负责。
因此,在JDK 8及更新版本中,即使是代替永久代的元数据区,也会进行垃圾回收,但这是针对类的元数据,而不是针对普通对象的。
你知道哪些垃圾收集算法
GC 最基础的算法有三种: 标记-清除算法、复制算法、标记-压缩算法,我们常用的垃圾回收
器一般都采用分散收集算法。
标记-清除算法, "标记-清除"( Mark-Sweep)算法,如它的名字一样,算法分为"标记"
和"清除"两个阶段:首先标记出所有需要回收的对象,在标记完成后统一回收掉所有被标记
的对象。
复制算法, "复制"(Copying)的收集算法,它将可用内存按容量划分为大小相等的两块,每次只使用其中的一块。当这一块的内存用完了,就将还存活着的对象复制到另外一块上面,
然后再把已使用过的内存空间一次清理掉。
标记-压缩算法,标记过程仍然与"标记-清除"算法一样,但后续步骤不是直接对可回收对象进行
清理,而是让所有存活的对象都向一端移动,然后直接清理掉端边界以外的内存
分代收集算法, "分代收集"( Generational Collection)算法,把Java 堆分为新生代和老年代, 这样就可以根据各个年代的特点采用最适当的收集算法。
总结
本文介绍了四道面试中经常出现的八股文,希望对大家有所帮助。
往期文章
金三银四面试题(一):JVM类加载与垃圾回收
金三银四面试题(二):数据库缓存的数据一致性
金三银四面试题(三):JVM内存模型
金三银四面试题(四):Full GC 和 Minor GC
金三银四面试题(五):JVM之TLAB
金三银四面试题(六):对象大小知多少