JVM常见面试题

版权声明:本文为博主原创文章,遵循 CC 4.0 by-sa 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/u012485165/article/details/81079145

 

1.说一下jdk的对空间的内存划分是怎样的?

Jdk1.7堆空间划分如下

Jdk1.8堆空间将永久代取消,改为元空间

2.GC的回收流程是怎样的?
GC回收流程如下:

对于整个的GC流程里面,那么最需要处理的就是新生代和老年代的内存清理操作,而元空间(永久代)都不在GC范围内

  1. 当现在有一个新的对象产生,那么对象一定需要内存空间,平均每个栈内存存4k,每个堆内存存8k,那么对象一定需要进行堆空间的申请

  2. 首先会判断Eden区是否有内存空间,如果此时有内存空间,则直接将新对象保存在伊甸园区。

  3. 但是如果此时在伊甸园区内存不足,那么会自动执行一个Minor GC 操作,将伊甸园区的无用内存空间进行清理,Minor GC的清理范围只在Eden园区,清理之后会继续判断Eden园区的内存空间是否充足?如果内存空间充足,则将新对象直接在Eden园区进行空间分配。

  4. 如果执行Minor GC 之后发现伊甸园区的内存空间依然不足,那么这个时候会执行存活区的判断,如果存活区有剩余空间,则将Eden园区部分活跃对象保存在存活区,那么随后继续判断Eden园区的内存空间是否充足,如果充足怎则将新对象直接在Eden园区进行空间分配。

  5. 此时如果存活区没有内存空间,则继续判断老年区。则将部分存活对象保存在老年代,而后存活区将有空余空间。

  6. 如果这个时候老年代也满了,那么这个时候将产生Major GC(Full GC),那么这个时候将进行老年代的清理

  7. 如果老年代执行Full GC之后,无法进行对象的保存,则会产生OOM异常,OutOfMemoryError异常

3.请解释StackOverflowError和OutOfMemeryError的区别?

通过之前的分析可以发现,实际上每一块内存中都会存在有一部分的可变伸缩区,其基本流程为:如果空间内存不足,在可变范围之内扩大内存空间,当一段时间之后发现内存充足,会缩小内存空间。

永久代(JDK 1.8后消失了)

虽然java的版本是JDK1.8,但是java EE 的版本还是jdk1.7,永久代存在于堆内存之中

元空间

元空间在Jdk1.8之后才有的,器功能实际上和永久代没区别,唯一的区别在于永久代使用的是JVM的堆内存空间,元空间使用的是物理内存,所以元空间的大小受本地内存影响,一般默认在2M 左右。

范例:设置一些参数,让元空间出错

Java -XX:MetaspaceSize=1m

4.JVM的引用类型有哪些?

引用内型:

强引用:当内存不足的时候,JVM宁可出现OutOfMemoryError错误停止,也需要进行保存,并且不会将此空间回收。在引用期间和栈有联系就无法被回收

软引用:当内存不足的时候,进行对象的回收处理,往往用于高速缓存中;mybatis就是其中

弱引用:不管内存是否紧张,只要有垃圾了就立即回收

幽灵引用:和没有引用是一样的

5.说说垃圾回收期的一些常见算法?

GC算法:

  1. 引用计数法:对于一个 对象A,只要引用一次就加1,引用失效减一,当计数为0时就失效

  2. 根搜索算法:判断是否可达:需要和根节点有依赖关系。如果没有和我的GCroots有关

 

  1. 标记-清除

标记-清除分为两个阶段:标记和清除,从根节点对对象就行标记,从根节点开始可达的对象就标记,清除阶段 清除未被标记的对象

  1. 标记-压缩

标记出对象是否存活,移动存活对象,移动完之后,清理边界外的对象

标记压缩对标记清除而言,有什么优势呢?

  1. 复制算法

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。即接口和实现类的关系

  1. 请解释StackOverflowError和OutOfMemeryError的区别?

答:StackOverflowError栈溢出,一般由于递归过多,调用方法过多导致

OutOfMemeryError堆内存溢出,即OOM,由于堆内存中没有被GC回收的对象过多导致。

出现OOM的原因:

(-).Java虚拟机的堆内存设置不够,可以通过参数-Xms和-Xmx来调优

(二)程序中创建了大量对象,并且长时间不能被被垃圾回收器回收(存在引用)

7.JVM的常用参数调优你知道哪些?

两个:一个-Xms堆内存初始化大小,一般默认为物理内存的六十四分之一

一个-Xmx堆内存最大分配空间,一般默认为物理内存的四分之一

8.内存快照抓取和MAT分析DUMP文件知道吗?

9.谈谈JVM中,对类加载器的认识

答:类加载器是JVM的组成部分之一。将字节码文件加载进JVM。

类加载分为四部分:

BootStrapClassLoader,即跟类加载器,加载java运行时所需的类,如String,Integer等存在${java_home}/jre/lib/rt.jar包类的所有类。

ExtensionClassLoader,扩展类加载器,加载一些扩展类,即${java_home}/jre/lib/ext/*.jar包

AppClassLoader,系统加载类,加载自定义的类,级classpath下的所有类

ClassLoader 抽象类加载器:用户自定义的类加载器,用户定义的类加载器都要继承次ClassLoader

Jvm默认采用的是双亲委派类加载机制,即先加载父类在加载子类,对上面四个类加载器采用自顶向下加载

 

 

10.在JVM中,如何判断一个对象是否死亡?

    判断对象是否死亡有两种方法:1.引用计数法,2.可达性分析算法

            引用计数法是最简单最古老的算法,JVM为每个对象分配一个计数器,当对象被引用时,计数器就加1,当对象没有被引用或者离开作用域,计数器就减1。当计数器的值为0时,就代表该对象已经死亡

             可达性分析算法,是用GCROOTs 作为对象的起点开始往下搜索,能搜索到这个对象,就表示对象是可达的,不能搜素到表示对象是不可达的

            可作为GCRoots的对象包括下面几种:

                        虚拟机栈中引用的对象

                        方法区中类静态属性引用的对象

                        方法区中常量引用的对象

                        本地方法栈中引用的对象

11.举例几种常见的垃圾收集算法?

        1.标记清除法

                标记清除法用在新生代,首先标记出所需要回收的对象,在标记完成后统一回收所有被标记的对象

        2.复制算法

                复制算法用在存活区,把不需要回收的对象全部放在s0/s1区,然后将s1/s0区进行统一的清理

         3.标记整理法

                标记整理法用在老年代,将需要回收的对象压缩在一边进行垃圾回收

展开阅读全文

没有更多推荐了,返回首页