年底了来一篇实用的硬核文章。本篇文章多达38道面试题,照顾到了JVM的方方面面,都是常见的题目。如果背诵记忆下来,进入大厂非常的easy。
面试题不能坑人,所以本篇文章的内容是经过多次打磨的,现在放送给大家。
有些面试题是开放性的,有些是知识性的,注意区别。面试并没有标准答案,尤其是开放性题目,你需要整理成白话文,来尽量的展示自己。
如果你在答案中描述了一些自己不是很熟悉的内容,可能会受到追问。所以,根据问题,整理一份适合自己的吧,这比拿来主义更让人印象深刻。
1、JVM有哪些内存区域?(JVM的内存布局是什么?)
JVM包含堆、元空间、Java虚拟机栈、本地方法栈、程序计数器等内存区域。其中,堆是占用内存最大的一块。我们平常的-Xmx、-Xms等参数,就是针对于堆进行设计的。
- 堆:JVM堆中的数据,是共享的,是占用内存最大的一块区域
- 虚拟机栈:Java虚拟机栈,是基于线程的,用来服务字节码指令的运行
- 程序计数器:当前线程所执行的字节码的行号指示器
- 元空间:方法区就在这里,非堆本地内存:其他的内存占用空间
2、Java的内存模型是什么?(JMM是什么?)
JVM试图定义一种统一的内存模型,能将各种底层硬件及操作系统的内存访问差异进行封装,使Java程序在不同硬件及操作系统上都能达到相同的并发效果。它分为工作内存和主内存,线程无法对主存储器直接进行操作,一个线程要和另外一个线程通信,只能通过主存进行交换。
JMM可以说是Java并发的基础,它的定义将直接影响多线程实现的机制,如果你想要想深入了解多线程并发中的相关问题现象,对JMM的深入研究是必不可少的。
上面两个问题是经常容易搞混的,但它们的内容却完全不同的。
3、JVM垃圾回收时候如何确定垃圾?什么是GC Roots?
JVM采用的是可达性分析算法。JVM是通过GC Roots来判定对象的存活的。从GC Roots向下追溯、搜索,会产生一个叫做Reference Chain的链条。当一个对象不能和任何一个GC Root产生关系,就判定为垃圾。
GC Roots大体包括:
- 活动线程相关的各种引用,比如虚拟机栈中栈帧里的引用。
- 类的静态变量的引用。
- JNI引用等。
当然也有比较详细的回答,个人认为这些就够了。详细版本如下:
- Java线程中,当前所有正在被调用的方法的引用类型参数、局部变量、临时值等。也就是与我们栈帧相关的各种引用。
- 所有当前被加载的Java类。
- Java类的引用类型静态变量。
- 运行时常量池里的引用类型常量(String或Class类型)。
- JVM内部数据结构的一些引用,比如sun.jvm.hotspot.memory.Universe类。
- 用于同步的监控对象,比如调用了对象的wait()方法。
- JNI handles,包括global handles和local handles
4、能够找到 Reference Chain 的对象,就一定会存活么?
这不一定,还要看reference类型。弱引用会在GC时会被回收,软引用会在内存不足的时候被回收。但没有Reference Chain的对象就一定会被回收。
5、强引用、软引用、弱引用、虚引用是什么?
普通的对象引用关系就是强引用。
软引用用于维护一些可有可无的对象。只有在内存不足时,系统则会回收软引用对象,如果回收了软引用对象之后仍然没有足够的内存,才会抛出内存溢出异常。
弱引用对象相比较软引用,要更加无用一些,它拥有更短的生命周期。当JVM进行垃圾回收时,无论内存是否充足,都会回收被弱