概述:知识点汇总
jvm的知识点汇总共6个大方向:内存模型、类加载机制、GC垃圾回收是比较重点的内容。性能调优部分偏重实际应用,重点突出实践能力。编译器优化和执行模式部分偏重理论基础,主要掌握知识点。前端
各个部分的内容以下:java
1>内存模型部分:程序计数器、方法区、堆、栈、本地方法栈的做用,保存哪些数据;面试
2>类加载部分:双亲委派的加载机制以及经常使用类加载器分别加载哪一种类型的类;算法
3>GC部分:分代回收的思想和依据,以及不一样垃圾回收算法实现的思路、适合的场景;编程
4>性能调优部分:经常使用的jvm优化参数的做用,参数调优的依据,要了解经常使用的jvm分析工具能分析哪类问题,以及使用方法;bootstrap
5>执行模式部分:解释、编译、混合模式的优缺点,了解java7提供的分层编译技术。须要知道JIT即时编译技术和OSR也就是栈上替换,知道C一、C2编译器针对的场景,其中C2针对server模式,优化更激进。在新技术方面能够了解一下java10提供的由java实现的graal编译器。安全
6>编译优化部分:前端编译器javac的编译过程、AST抽象语法树、编译期优化和运行期优化。编译优化的经常使用技术,包括公共子表达式的消除、方法内联、逃逸分析、栈上分配、同步消除等。明白了这些才能写出对编译器友好的代码。多线程
jvm的内容相对来讲比较集中,可是对知识深度的掌握要求较高,建议面试前重点增强。并发
1、jvm内存相关考点
1.详解-jvm内存模型 app
jvm内存模型主要指运行时的数据区,包括5个部分。
栈也叫方法栈,是线程私有的,线程在执行每一个方法时都会同时建立一个栈帧,用来存储局部变量表、操做栈、动态连接、方法出口等信息。调用方法时执行入栈,方法返回时执行出栈。
本地方法栈与栈相似,也是用来保存线程执行方法时的信息,不一样的是,执行java方法使用栈,而执行native方法使用本地方法栈。
程序计数器保存着当前线程所执行的字节码位置,每一个线程工做时都有一个独立的计数器。程序计数器为执行java方法服务,执行native方法时,程序计数器为空。
栈、本地方法栈、程序计数器这三个部分都是线程独占的。
堆是jvm管理的内存中最大的一块,堆被全部线程共享,目的是为了存放对象实例,几乎全部的对象实例都在这里分配。当堆内存没有可用的空间时,会抛出OOM异常。根据对象存活的周期不一样,jvm把堆内存进行分代管理,由垃圾回收器来进行对象的回收管理。
方法区也是各个线程共享的内存区域,又叫非堆区。用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据,
jdk1.7中的永久代和1.8中的metaspace都是方法区的一种实现。
面试回答此知识点相关问题时,要答出两个要点:一个是各部分的功能,另外一个是哪些线程共享,哪些独占。
2.详解-jmm内存可见性
jmm是java内存模型,与刚才讲到的jvm内存模型是两回事,jmm的主要目标是定义程序中变量的访问规则,如图所示,全部的共享变量都存储在主内存中共享。每一个线程有本身的工做内存,工做内存中保存的是主内存中变量的副本,线程对变量的读写等操做必须在本身 的工做内存中进行,而不能直接读写主内存中的变量。
在多线程进行数据交互时,例如线程a给一个共享变量赋值后,由线程b来读取这个值,a修改完变量是修改在本身的工做区内存中,b是不可见的,只有从a的工做区写回到主内存,b再从主内存读取到本身的工做区才能进行进一步的操做。因为指令重排序的存在,这个写-读的顺序有可能被打乱。
所以jmm须要提供原子性、可见性、有序性的保证。
三、详解-jmm保证
主要介绍下jmm如何保证原子性、可见性,有序性。
jmm保证对除long和double外的基础数据类型的读写操做是原子性的。另外关键字Synchronized也能够提供原子性保证。Synchronized的原子性是经过java的两个高级的字节码指令monitorenter和monitorexit来保证的。
jmm可见性的保证,一个是经过Synchronized,另一个就是volatile。volatile强制变量的赋值会同步刷新回主内存,强制变量的读取会从主内存从新加载,保证不一样的线程老是可以看到该变量的最新值。
jmm对有序性的保证,主要经过volatile和一系列happens-before原则。volatile的另外一个做用就是阻止指令重排序,这样就能够保证变量读写的有序性。
happens-before原则包括一系列规则,如
程序顺序原则,即一个线程内必须保证语义串行性;
锁规则,即对同一个锁的解锁必定发生在再次加锁以前;
此外还包括happens-before原则的传递性、线程启动、中断、终止规则等。
2、类加载机制相关考点
1.详解类加载机制
类的加载指的是将编译好的class类文件中的字节码读入到内存中,将其放在方法区内并建立对应的Class对象。
类的加载分为加载、连接、初始化,其中连接又包括验证、准备、解析三步。看到图中上半部分深绿色,咱们逐个分析:
加载是文件到内存的过程。经过类的彻底限定名查找此类字节码文件,并利用字节码文件建立一个Class对象
验证是对类文件内容验证。目的在于确保Class文件符合当前虚拟机要求,不会危害虚拟机自身安全。主要包括四种:文件格式验证,元数据验证,字节码验证,符号引用验证。
准备阶段是进行内存分配。为类变量也就是类中由static修饰的变量分配内存,而且设置初始值,这里要注意,初始值是0或者null,而不是代码中设置的具体值,代码中设置的值是在初始化阶段完成的。另外这里也不包含用final修饰的静态变量,由于final在编译的时候就会分配了。
解析主要是解析字段、接口、方法。主要是将常量池中的符号引用替换为直接引用的过程。直接引用就是直接指向目标的指针、相对偏移量等。
最后是初始化:主要完成静态块执行与静态变量的赋值。这是类加载最后阶段,若被加载类的父类没有初始化,则先对父类进行初始化。
只有对类主动使用时,才会进行初始化,初始化的触发条件包括建立类的实例的时候、访问类的静态方法或者静态变量的时候、Class.forName()反射类的时候、或者某个子类被初始化的时候。
类的生命周期,就是从类的加载到类实例的建立与使用,再到类对象再也不被使用时能够被GC卸载回收。这里要注意一点,由java虚拟机自带的三种类加载器加载的类在虚拟机的整个生命周期中是不会被卸载的,只有用户自定义的类加载器所加载的类才能够被卸载。
2.详解类加载器
java自带的三种类加载器分别是:bootstrap启动类加载器、扩展类加载器和应用加载器也叫系统加载器。图右边的桔黄色文字表示各种加载器对应的加载目录。启动类加载器加载java home中lib目录下的类,扩展加载器负责加载ext目录下的类,应用加载器加载classpath指定目录下的类。
除此以外,能够自定义类加载器。
java的类加载使用双亲委派模式,即一个类加载器在加载类时,先把这个请求委托给本身的父类加载器去执行,若是父类加载器还存在父类加载器,就继续向上委托,直到顶层的启动类加载器,如图中蓝色向上的箭头。若是父类加载器可以完成类加载,就成功返回,若是父类加载器没法完成加载,那么子加载器才会尝试本身去加载。
这种双亲委派模式的好处,一个能够避免类的重复加载,另外也避免了java的核心API被篡改。
3、其余知识梳理
1.详解分代回收
前面提到过,java的堆内存被分代管理,分代管理主要是为了方便垃圾回收,这样作基于2个事实,第1、大部分对象很快就再也不使用,第二,还有一部分不会当即无用,但也不会持续很长时间。
虚拟机中划分为年轻代、老年代、和永久代。
1>年轻代:主要用来存放新建立的对象,年轻代分为eden区和两个Survivor区。大部分对象在Eden区中生成。当Eden区满时,还存活的对象会在两个Survivor区交替保存,达到必定次数的对象会晋升到老年代。
2>老年代:用来存放从年轻代晋升而来的,存活时间较长的对象。
3>永久代:主要保存类信息等内容,这里的永久代是指对象划分方式,不是专指1.7的permGen,或者1.8以后的metaspace。
根据年轻代与老年代的特色,jvm提供了不一样的垃圾回收算法。垃圾回收算法按类型能够分为引用计数法、复制法和标记清除法。
其中引用计数法是经过对象被引用的次数来肯定对象是否被使用,缺点是没法解决循环引用的问题。
复制算法须要from和to两块相同大小的内存空间,对象分配时只在from块中进行,回收时把存活对象复制到to块中,并清空from块,而后交换两块的分工,即把from块做为to块,把to块做为from块。缺点是内存使用率较低。
标记清除算法分为标记对象和清除不在使用的对象两个阶段,标记清除算法的缺点是会产生内存碎片。
jvm中提供的年轻代回收算法Serial、ParNew、Parallel Scavenge都是复制算法,而CMS、G一、zgc都属于标记清除算法。
本篇文章,对这几个算法就不展开了,具体可见《32个Java面试必考点》
总结:面试考察点及加分项
1.jvm相关的面试考察点
首先,须要jvm的内存模型和java的内存模型;
其次,要了解的类的加载过程,了解双亲委派机制;
第三,要理解内存的可见性与java内存模型对原子性、可见性、有序性的保证机制;
第四,要了解经常使用的gc算法的特色、执行过程,和适用场景,例如g1适合对最大延迟有要求的场合,zgc适用于64为系统的大内存服务中;
第五,要了解经常使用的jvm参数,明白对不一样参数的调整会有怎样的影响,适用什么样的场景。例如垃圾回收的并发数、偏向锁设置等
2.相关加分项
若是想要面试官对你留下更好的印象的话,注意这些加分项:
首先,若是在编译器优化方面有深刻的了解的话,会让面试官以为你对技术的深度比较有追求。例如知道在编程时如何合理利用栈上分配下降gc压力、如何编写适合内联优化等代码等。
其次,若是你能有线上实际问题的排查经验或思路那就更好了,面试官都喜欢动手能力强的同窗。例如解决过线上常常full gc问题,排查过内存泄露问题等。
第三,若是能有针对特定场景的jvm优化实践或者优化思路,也会有意想不到的效果。例如针对高并发低延迟的场景,如何调整gc参数尽可能下降gc停顿时间,针对队列处理机如何尽量提升吞吐率等;
第四,若是对最新的jvm技术趋势有所了解,也会给面试官留下比较深入的印象。例如了解zgc高效的实现原理,了解Graalvm的特色等。
总之,掌握以上具体的JVM考点,才能在面试时应答自如。但愿读完此篇文章的你,都能在金三银四的招聘季作好准备,拿到心仪的Offer。
以上内容摘取自《32个Java面试必考点》第03讲:深刻浅出JVM,点此查看更多