面试中,JVM属于必问点。对于工作经验在3年左右的初级开发而言,如果你照着本文给出的模式,有条理地回答出这方面的问题,那么面试官一定会认为你很资深。
先来看下JVM的基础问题。
1.说一下jdk的对空间的内存划分是怎样的?
Jdk1.7堆空间划分如下
Jdk1.8堆空间将永久代取消,改为元空间
2.GC的回收流程是怎样的?
对于整个的GC流程里面,那么最需要处理的就是新生代和老年代的内存清理操作,而元空间(永久代)都不在GC范围内
当现在有一个新的对象产生,那么对象一定需要内存空间,平均每个栈内存存4k,每个堆内存存8k,那么对象一定需要进行堆空间的申请
首先会判断Eden区是否有内存空间,如果此时有内存空间,则直接将新对象保存在伊甸园区。
但是如果此时在伊甸园区内存不足,那么会自动执行一个Minor GC 操作,将伊甸园区的无用内存空间进行清理,Minor GC的清理范围只在Eden园区,清理之后会继续判断Eden园区的内存空间是否充足?如果内存空间充足,则将新对象直接在Eden园区进行空间分配。
如果执行Minor GC 之后发现伊甸园区的内存空间依然不足,那么这个时候会执行存活区的判断,如果存活区有剩余空间,则将Eden园区部分活跃对象保存在存活区,那么随后继续判断Eden园区的内存空间是否充足,如果充足怎则将新对象直接在Eden园区进行空间分配。
此时如果存活区没有内存空间,则继续判断老年区。则将部分存活对象保存在老年代,而后存活区将有空余空间。
如果这个时候老年代也满了,那么这个时候将产生Major GC(Full GC),那么这个时候将进行老年代的清理
如果老年代执行Full GC之后,无法进行对象的保存,则会产生OOM异常,OutOfMemoryError异常
3.请解释StackOverflowError和OutOfMemeryError的区别?
通过之前的分析可以发现,实际上每一块内存中都会存在有一部分的可变伸缩区,其基本流程为:如果空间内存不足,在可变范围之内扩大内存空间,当一段时间之后发现内存充足,会缩小内存空间。
虽然java的版本是JDK1.8,但是java EE 的版本还是jdk1.7,永久代存在于堆内存之中
元空间
元空间在Jdk1.8之后才有的,器功能实际上和永久代没区别,唯一的区别在于永久代使用的是JVM的堆内存空间,元空间使用的是物理内存,所以元空间的大小受本地内存影响,一般默认在2M 左右。
范例:设置一些参数,让元空间出错
Java -XX:MetaspaceSize=1m
4.JVM的引用类型有哪些?
引用内型:
强引用:当内存不足的时候,JVM宁可出现OutOfMemoryError错误停止,也需要进行保存,并且不会将此空间回收。在引用期间和栈有联系就无法被回收
软引用:当内存不足的时候,进行对象的回收处理,往往用于高速缓存中;mybatis就是其中
弱引用:不管内存是否紧张,只要有垃圾了就立即回收
幽灵引用:和没有引用是一样的
5.说说垃圾回收期的一些常见算法?
GC算法:
引用计数法:对于一个 对象A,只要引用一次就加1,引用失效减一,当计数为0时就失效
根搜索算法:判断是否可达:需要和根节点有依赖关系。如果没有和我的GCroots有关
标记-清除
标记-清除分为两个阶段:标记和清除,从根节点对对象就行标记,从根节点开始可达的对象就标记,清除阶段 清除未被标记的对象
标记-压缩
标记出对象是否存活,移动存活对象,移动完之后,清理边界外的对象
标记压缩对标记清除而言,有什么优势呢?
复制算法
6.请你谈谈你对JVM的理解?Java8的虚拟机有什么更新
答:在这里我就不说JVM是java虚拟机了。先谈谈java文件的运行过程一个*.java文件通过编译器编译成字节码文件,然后由JVM的类加载字节码文件,由执行引擎执行。在这期间类加载器加载的数据会与java运行时数据区进行交互。在这里看来JVM内存划分为类加载器,执行引擎,本地方法接口,java运行时数据区。其中java运行时数据区划分为:程序计数器,虚拟机栈,本地方法栈,堆,方法区
程序计数器:指向当前正在执行线程的字节码地址的指令或行号。属于线程私有的。
虚拟机栈:存储当前线程运行方法所需的数据指令,访问地址(java独有的)
Java虚拟机栈划分为四个部分:局部变量表,操作数栈,动态链接,方法出口
本地方法栈:存储本地方法
方法区:方法区包含静态变量+类信息+字面量常量+运行时常量池
堆:对象的新建都在堆区完成,JAVA中的GC主要操作的是这个区域
堆区在逻辑上划分为:新生代,老年代,永久代(java7),在java8将堆区在逻辑上划分为:新生代,老年代和元空间(Meta Space),元空间默认大小为2M左右,其中新生代又分为Eden区,存活区,存活区由;两块大小相等的内存空间组成(即s0区,s1区),也叫from区和to区,JVM在的堆区在物理上只有两块区:新生代和老年代。Java8的虚拟机的更新:元空间将永久代取代,为什么会更新呢?Oracle整合了Sun公司的Hotspot和BEA公司的Jrocket,从而将永久代去掉用元空间取代,永久代和元空间有什么区别?元空间在Jdk1.8之后才有的,其功能实际上和永久代没区别,唯一的区别在于永久代使用的是JVM的堆内存空间,元空间使用的是物理内存,所以元空间的大小受本地内存影响。
在来谈谈方法区和永久代的关系:方法区相当去定义了UserService,永久代相当去UserServiceImpl。即接口和实现类的关系
请解释StackOverflowError和OutOfMemeryError的区别?
答:StackOverflowError栈溢出,一般由于递归过多,调用方法过多导致
OutOfMemeryError堆内存溢出,即OOM,由于堆内存中没有被GC回收的对象过多导致。
7.JVM的常用参数调优你知道哪些?
两个:一个-Xms堆内存初始化大小,一般默认为物理内存的六十四分之一
一个-Xmx堆内存最大分配空间,一般默认为物理内存的四分之一
然后当你回答好任何一个JVM问题时,再扩展会说下:在项目里,我会在编码时注意JVM的效率,更有OOM问题的排查经验。
8 如何在编码时注意JVM效率?
1 String等不可变类,尽量避免频繁读写,比如要频繁读写String,就要用StringBuild或StringBuffer
2 不写finalize方法,避免因写错该方法而导致的对象不可被回收。
3 必要时用到弱和软引用。
4 用好ArrayList等对象立即clear,用好Connection等对象立即close。
9 出现OOM问题时如何排查?
1 看出现问题时的dump文件,看当时的内存镜像。
2 看当时的日志文件
结合上述两点,一般能看出问题所在,比如物理对象没有释放等。
10 你结果过哪些OOM的问题?
1 ArrayList等对象用好没释放。
2 物理对象用好没释放
3 在多线程里,频繁创建对象