面试时如何通过JVM展示自身实力

面试中,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 在多线程里,频繁创建对象

猜你喜欢

转载自blog.csdn.net/sxeric/article/details/115303187
今日推荐