- 架构图
(JVM-HotSpot虚拟机整体架构图)
(JVM 运行时数据区架构图)
(对象内存分配流程图)
- 面试题解答与总结
2.1 (京东)JVM内存结构,Eden和Surviver比例? JVM运行时内存分区:方法区,堆,虚拟机栈,程序计数器,本地方法栈。 堆区中年轻代包括一个Eden区和两个Survivor区,默认比例是8:1:1。
2.2 (京东)JVM内存为什么要分为新生代、老年代和永久代(元空间)?新生代中为什么要分为Eden和Survivor? 根本原因:内存分区的根本目的是为了优化GC效率; 进一步解释如何优化GC性能: 程序中超过80%的对象都是用完就会释放掉的,内存分代使绝大部分GC发生在新生代,很少会出现老年代GC。具体来说一方面使GC发生在特定的新生代区间,另一方面提高GC的垃圾命中率。
2.3(天猫)JVM内存模型以及分区?需要详细到每个区放什么? JVM内存模型是指JVM架构中的运行时数据区。 分为五个区: 方法区:类信息、域信息、方法信息,运行时常量池 堆:所有对象实例 虚拟机栈:以方法为单位的栈帧(局部变量表、操作数栈) 程序计数器:存储当前线程的现场信息:即正在执行的字节码的地址和行号 本地方法栈:本地方法接口声明
2.4 (天猫)JVM 内存模型,java8做了什么修改? 废弃永久代,改为元空间; 永久代存在于堆内存中,元空间存在于系统本地内存,取消了永久代的大小限制。 元空间和永久代类似,是方法区的实现方式。 补充:java7的变化是静态变量和字符串常量池从方法区移动到堆内存; 而最新存在于本地内存的元空间存储的内容包括:类信息,域信息,方法信息,运行时常量池。
2.5 (拼多多)JVM内存分哪几个区?每个区的作用是什么? JVM本质上是由指令、数据、控制组成一个操作系统,主要功能包括翻译和内存管理。 方法区:存储加载到内存中的字节码信息(类信息,域信息,方法信息,运行时常量池) 堆:存储所有运行时对象,作用是存储数据及内存管理 虚拟机栈:存储以方法为单位的栈帧,配合程序计数器进行程序执行控制 程序计数器:存储线程当前执行的字节码地址和行号,程序执行控制 本地方法栈:和系统底层交互,使用系统提供的基础能力。
2.6 (美团)java 内存分配 首先,阐述JVM运行时内存区的五个分区:如上; 然后,阐述对象分配和GC过程: 新创建对象在Eden区分配,Eden区满了之后触发新生代GC回收掉Eden区大部分垃圾对象,少部分未回收的对象晋级到Survivor的to区。再次触发GC时上次未被回收的对象会从form区转移到to区,两个survivor区的每次转移都会有一个是空的即to区,对象每次进入to区都会计数+1,默认增加到15就会在当次GC时晋升到老年代。当老年代区间满时,会触发老年代GC,同时对堆空间和方法区进行垃圾回收。 (堆中新生代和老年代默认比例为1:2)
2.7 (美团)JVM的永久代中会发生垃圾回收吗? 会发生。 永久代jdk1.8之后称作元空间,元空间和永久代类似,都是方法区的实现方式。 老年代空间满时会触发MajorGC,会对堆和方法区进行全面垃圾回收,方法区的垃圾回收比较基类,对于类型和静态常量池中的垃圾判断条件都很严格。
2.8 (蚂蚁金服) JVM内存分布/内存结构?栈和堆的区别?堆的结构?为什么两个survivor区? 内存结构:方法区、堆、虚拟机栈、程序计数器、本地方法栈 栈和堆的区别:栈是连续空间,效率高,不需要垃圾回收;堆是非连续空间,效率相对较低,需要垃圾回收; 堆的结构:年轻代(一个Eden区,两个survivor区,默认比理8:1:1),老年代,永久代/元空间 为什么两个survivor区:优化GC效率,年轻代GC时用于计数未被回收对象的年龄。
- 9 什么时候对象会进入老年代?
新生代中年龄超过15岁的对象; 大对象分配时,新生代GC后仍无法满足大对象内存空间需要的情况下,会直接在老年代分配对象内存。
参考: https://www.bilibili.com/video/BV1PJ411n7xZ?p=1&share_medium=android&share_plat=android&share_session_id=29126549-f459-496d-a636-f5f7897d01b5&share_source=WEIXIN&share_tag=s_i×tamp=1630375445&unique_k=C5jeW2
https://zhuanlan.zhihu.com/p/74772956
|