八股文之JVM

前言

此处主要用于总结面试中出现的JVM相关的题目。

所属:八股文之JAVA基础

大纲

JVM 类加载 类字节码 类文件结构 常量池 访问标志 类和接口索引 字段表属性 方法表属性 属性表属性 理解字节码 javac,javap 其他语言,比如kotlin 类声明周期 加载 连接 验证:确保被加载的正确性 准备:为类的静态变量分配内存,并将其初始化为默认值 解析:把类的符号引用转换为直接引用 类加载周期 类加载的层次 启动类加载器 扩展类加载器 应用程序类加载器 自定义类加载器 寻找类加载器 类的加载 命令行启动应用程序由jvm初始化加载 通过Class.forName方法动态加载 通过ClassLoader.loadClass方法动态加载 jvm类加载机制 全盘负责 父类委托 缓存机制 双亲委派机制 初始化 使用 卸载 内存结构 程序计数器 Java虚拟机栈 本地方法栈 堆 新生代Young_Generation Eden伊利园 From_Survivor To_Survivor 老年代Old_Gernation 方法区 直接内存 元数据区内存 垃圾回收 判断一个对象是否可被回收 引用类型 常见的GC算法 新一代GC算法 垃圾收集器 内存分配与回收策略 GC日志解读与分析 Java内存模型 调优和排错 对象模型OOP-Klass 执行引擎 常见面试题

总结

JVM是Java运行基础,面试时一定会遇到JVM的有关问题,内容相对集中,但对只是深度要求较高。

其中内存模型,类加载机制,GC是重点方面。性能调优部分更偏向应用,重点突出实战能力。

编译器优化和执行模式部分偏向于理论基础。

重点掌握知识点:

  • 需了解内存模型各部分作用,保存哪些数据;
  • 类加载双亲委派加载机制,常见加载器分别加载哪种类型的类;
  • GC分代回收的思想和依据以及不同垃圾回收算法的回收思路和适合场景;
  • 性能调优常有JVM优化参数作用,参数调优的依据,常见的JVM分析工具能分析哪些问题以及使用方法;
  • 执行模式解释/编译/混合模式的优缺点,Java7提供的分层编译技术,JIT即时编译技术,QSR栈上替换,C1/C2编译器针对的场景,C2针对的是server模式,优化更激进。新技术方面Java10的graal编译器
  • 编译器优化javac的编译过程,ast抽象语法树,编译器优化和运行器优化。

七、JVM

3、说说类加载与卸载/知道类的生命周期吗?

类加载过程主要分为7个步骤:加载、验证、准备、解析、初始化、使用和卸载,其中验证、准备、解析合称连接。
加载 通过类的完全限定名,查找此类字节码文件,利用字节码文件创建Class对象。
验证 确保Class文件符合当前虚拟机的要求,不会危害到虚拟机自身安全。
准备 进行内存分配,为static修饰的类变量分配内存,并设置初始值(0或者null),不包含final修饰的静态变量,因为final变量在编译时分配;
解析 将常量池中的符号引用替换为直接引用的过程,直接引用为直接指向目标的指针或者相对偏移量等;
初始化主要完成静态块执行以及静态变量的赋值,先初始化父类,再初始化当前类,只有对类主动使用时才会初始化;

触发条件包括,创建类的实例时,访问类的静态方法或静态变量的时候,使用Class.forName反射类的时候,或者某个子类初始化的时候。

Java自带的加载器加载的类,在虚拟机的生命周期中是不会被卸载的,只有用户自定义的加载器加载的类才可以被卸载。

1、加载机制-双亲委派机制

亲委派模式,即加载器加载类时先把请求委托给自己的父类加载器执行,直到顶层的启动类加载器,父类加载器能够完成加载则成功返回,

不能则子类加载器才自己尝试加载。
优点:

  • 避免了类的重复加载;
  • 避免Java的核心API被篡改

2、分代回收

分代回收基于两个事实:大部分对象很快就不使用了,还有一部分不会立即无用,但也不会持续很长时间。

年轻代->标记-复制

老年代->标记-清除

3、回收算法

a、G1算法

1.9后默认的垃圾回收算法,特点保持高回收率的同时减少停顿,采用每次只清理一部分,而不是清理全部的增量式清理,以保证停顿时间不会过长

其取消了年轻代与老年代的物理划分,但仍属于分代收集器,算法将堆分为若干个逻辑区域(region),一部分用作年轻代,一部分用作老年代,还有用来存储巨型对象的分区。

同CMS相同,会遍历所有对象,标记引用情况,清除对象后会对区域进行复制移动,以整合碎片空间。

年轻代回收:并行复制采用复制算法,并行收集,会StopTheWorld。

老年代回收: 会对年轻代一并回收。

初始标记完成堆root对象的标记,会StopTheWorld。 并发标记 GC线程和应用线程并发执行, 最终标记完成三色标记周期,会StopTheWorld。 复制/清楚会优先对可回收空间加大的区域进行回收。

b、ZGC算法

前面提供的高效垃圾回收算法,针对大堆内存设计,可以处理TB级别的堆,可以做到10ms以下的回收停顿时间。

  • 着色指针
  • 读屏障
  • 并发处理
  • 基于region
  • 内存压缩(整理)

roots标记:标记root对象,会StopTheWorld。

并发标记:利用读屏障与应用线程一起运行标记,可能会发生StopTheWorld。清除会清理标记为不可用的对象。

roots重定位:是对存活的对象进行移动,以腾出大块内存空间,减少碎片产生.重定位最开始会StopTheWorld,却决于重定位集与对象总活动集的比例.。并发重定位与并发标记类似。

4、简述一下JVM的内存模型/Java内存结构

JVM定义了不同运行时数据区,他们是用来执行应用程序的。某些区域随着JVM启动及销毁,另外一些区域的数据是线程性独立的,随着线程创建和销毁。

JVM在执行Java程序时,会把它管理的内存划分为若干个的区域,每个区域都有自己的用途和创建销毁时间。如下图所示,可以分为两大部分,线程私有区和共享区。

JVM内存分为线程私有区和线程共享区:

线程私有区:程序计数器、虚拟机栈、本地方法栈

线程共享区:方法区、堆

5、说说堆和栈的区别

栈是运行时单位,代表着逻辑,内含基本数据类型和堆中对象引用,所在区域连续,没有碎片;

堆是存储单位,代表着数据,可被多个栈共享(包括成员中基本数据类型、引用和引用对象),所在区域不连续,会有碎片。

  • 功能不同:栈内存用来存储局部变量和方法调用,而堆内存用来存储Java中的对象。无论是成员变量,局部变量,还是类变量,它们指向的对象都存储在堆内存中。
  • 共享性不同:栈内存是线程私有的。 堆内存是所有线程共有的。
  • 异常错误不同:
    • 如果栈内存或者堆内存不足都会抛出异常。 栈空间不足:java.lang.StackOverFlowError。
    • 堆空间不足:java.lang.OutOfMemoryError。
  • 空间大小:栈的空间大小远远小于堆的。
6、什么时候会触发FullGC(感觉这个答案不是很好理解,后期更换)

除直接调用System.gc外,触发Full GC执行的情况有如下四种:

  1. 老年代空间不足:老年代空间只有在新生代对象转入及创建大对象、大数组时才会出现不足的现象,当执行Full GC后空间仍然不足,则抛出如下错误: java.lang.OutOfMemoryError: Java heap space 为避免以上两种i清华看引起的Full GC,调优时应尽量做到让对象在Minor GC阶段被回收、让对象在新生代多存活一段时间及不要创建过大的对象及数组。
  2. Permanet Generation空间满:Permanet Generation中存放的为一些class的信息等,当系统中要价值的类、反射的类和调用的方法较多时,Permanet Generation可能会被占满,在未配置为采用CMS GC的情况下会执行Full GC。如果经过Full GC仍然回收不了,那么JVM会抛出如下错误信息:java.lang.OutOfMemoryError: PermGem space。 为避免Perm Gen占满造成Full GC现象,可采用的方法为增大Perm Gen空间或转为使用CMS GC。
  3. CMS GC时出现promotion failed和concurrent mode failure 对于采用CMS进行旧生代GC的程序而言,尤其要注意GC日志中是否还有promotion failed和concurrent mode failure两种状况,当这两种状况出现时可能除触发Full GC。
  4. 统计得到的Minor GC普升到老年代的平均大小大于老年代的剩余空间 这是一个较为复杂的触发情况,Hotspot为了避免由于新生代对象晋升到旧生代导致旧生代空间不足的现象,在进行Minor GC时,做了一个判断,如果之前统计所得到的Minor GC晋升到旧生代的平均大小大于旧生代的剩余空间,那么就直接触发Full GC
7、什么是Java虚拟机?为什么Java被称作是"平台无关的编程语言"?

Java虚拟机是一个可以执行Java字节码的虚拟机进程。Java源文件被编译成能被Java虚拟机执行的字节码文件。 Java被设计成允许应用程序可以运行在任意的平台,而不需要程序员为每一个平台单独重写或者是重新编译。Java虚拟机让这个变为可能,因为它知道底层硬件平台的指令长度和其他特性。

9、说说对象分配规则
  • 对象优先分配在Eden区,如果Eden区没有足够的空间时,虚拟机执行一次Minor GC。
  • 大对象直接进入老年代(大对象是指需要大量连续内存空间的对象)。这样做的目的是避免在Eden区和两个Survivor区之间发生大量的内存拷贝(新生代采用复制算法收集内存)。
  • 长期存活的对象进入老年代。虚拟机为每个对象定义了一个年龄计数器,如果对象经过了1次Minor GC那么对象会进入Survivor区,之后每经过一次Minor GC那么对象的年龄加1,知道达到阀值对象进入老年区。
  • 动态判断对象的年龄。如果Survivor区中相同年龄的所有对象大小的总和大于Survivor空间的一半,年龄大于或等于该年龄的对象可以直接进入老年代。
  • 空间分配担保。每次进行Minor GC时,JVM会计算Survivor区移至老年区的对象的平均大小,如果这个值大于老年区的剩余值大小则进行一次Full GC,如果小于检查HandlePromotionFailure设置,如果true则只进行Monitor GC,如果false则进行Full GC。
10、描述一下JVM加载class文件的原理机制?

JVM中类的装载是由类加载器(ClassLoader)和它的子类来实现的,Java中类加载器是一个重要的Java运行时系统组件,它负责在运行时查找和装入类文件中的类。由于Java的跨平台性,经过编译的Java源程序并不是一个可执行程序,而是一个或多个类文件。当Java程序需要使用某个类时,JVM会确保这个类已经被加载、连接(验证、准备和解析)和初始化。类的加载是指把类的.class文件中的数据读入到内存中,通常是创建一个字节数组读入.class文件,然后产生与所家长类对应的Class对象。加载完成后,Class对象并不完整,所以此时的类还不可用。当类被加载后就进入连接阶段,这一阶段包括验证、准备(为静态变量分配内存并设置默认的初始值)和解析(将符号引用替换为直接引用)三个步骤。最后JVM对类进行初始化,包括:1)如果类存在直接的父类并且这个类还没有被初始化,那么就先初始化父类;2)如果类中存在初始化语句,就依次执行这些初始化语句。

类的加载是类加载器完成的,类加载器包括:根加载器(Bootstrap)、扩展加载器(Extension)、系统加载器(System)和用户自定义类加载器(java.lang.ClassLoader的子类)。

  • Bootstrap:一般用本地代码实现,负责加载JVM基础核心类库(rt.jar);
  • Extension:从java.ext.dirs系统属性所指定的目录中加载类库,它的父加载器是Bootstrap;
  • System:又叫应用类加载器,其父类是Extension。它是应用最广泛的类加载器。它从环境变量classpath或者系统属性java.class.path所指定的目录中记载类,是用户自定义加载器的默认父加载器。
11、说说Java对象创建过程

1、JVM遇到一条新建对象的指令时首先去检查这个指令的参数是否能在常量池中定义一个类的符号引用。然后加载这个类;

2、为对象分配内存。一种办法"指针碰撞"、一种办法"空闲列表",最终常用的办法"本地线程缓冲分配(TLAB)";

3、将除对象头外的对象内存空间初始化为0;

4、对对象头进行必要设置。

13、简述Java的对象结构(!!!)

Java对象由三个部分组成:对象头、实例数据、对象填充。

对象头由两部分组成,第一部分存储对象自身的运行时数据:哈希码、GC分代年龄、锁标识状态、线程持有的锁、偏向线程ID(一般占32/64 bit)。第二部分是指针类型,指向对象的类元数据类型(即对象代表哪个类)。如果是数组对象,则对象头中还有一部分用来记录数组长度。

实例数据用来存储对象真正的有效信息(包括父类继承下来的和自己定义的)

对齐填充:JVM要求对象起始地址必须是8字节的整数倍(8字节对齐)

28、对象头具体都包含哪些内容?

在我们常见的HotSpot虚拟机中,对象在内存中布局实际包含3个部分:

  1. 对象头
  2. 实例数据
  3. 对齐填充

而对象头包含两部分内容,Mark Word中的内容会随着锁标志位而发生变化,所以只说存储结构就好了。

  1. 对象自身运行时所需的数据,也被称为Mark Word,也就是用于轻量级锁和偏向锁的关键点。

  2. 存储类型指针,也就指向类的元数据的指针,通过这个指针才能确定对象是属于哪个类的实例。

如果是数组的话,则还包含了数组的长度。

14、如何判断对象可以被回收?

判断对象是否存活的一般有两种方式:

  • 引用技术:每个对象有一个引用技术属性,新增一个引用时技术加1,引用释放时计数减1,计数为0时可以回收。此方法简单,无法解决对象相互循环引用的问题。
  • 可达性分析:从GC Roots开始向下搜索,搜素所走过的路径称为引用链。当一个对象到GC Roots没有任何引用链相连时,则证明此对象是不可用的,不可达对象。
15、JVM的永久代中会发生垃圾回收么?

垃圾回收不会发生在永久代,如果永久代满了或者是超过了临界值,会触发完全垃圾回收(Full GC)。如果你仔细查看垃圾收集器的输出信息,就会发现永久代也是被回收的。这就是为什么正确的永久代大小对避免Full GC是非常重要的原因,请参与下java8:从永久代到元数据区(注:Java8中已经移除了永久代,新加了一个叫做元数据区的native内存区)

16、你知道哪些垃圾收集算法

GC最基础的算法有三种:标记-清除算法、复制算法、标记-压缩算法,我们常见的垃圾回收器一般都采用分代收集算法:

  • 标记-清除算法,"标记-清除"(Mark-Sweep)算法,如它的名字一样,算法分为"标记"和"清除"两个阶段:首先标记出所以后需要回收的对象,在标记完成后统一回收掉所以被标记的对象。
  • 复制算法。"复制"(Copying)的收集算法,它将可用内存按容量划分为大小相等的两块,每次只使用其中的一块。当这一块内存用完了,就将还存活的对象复制到另外一块上面,然后再把已使用过的内存空间一次清理掉。
  • 标记-压缩算法,标记过程仍然与"标记-清除"算法一样,但后续步骤不是直接对可回收对象进行清理,而是让所有存活的对象都向一端移动,然后直接清理掉端边界以外的内存。
  • 分代收集算法,"分代收集"(Generational Colleciton)算法,把Java堆分为新生代和老年代,这样就可以根据哥哥年代的特点采用最适当的收集算法。
17、调优命令有哪些?(!!!)

Sun JDK监控和故障处理命令有jps jstat jmap jhat jstack jinfo

  • jps,JVM Process Status Tool,显示指定系统内所有的HotSpot虚拟机进程。
  • jstat,JVM statistics Monitoring是用于监视虚拟机运行时状态信息的命令,它可以显示出虚拟机进程中的类装载、内存、垃圾收集、JIT编译等运行数据。
  • jmap,JVM Memory Map命令用于生成heap dump文件
  • jhat,JVM Heap Analysis Tool命令是与jmap搭配使用,用来分析jmap生成的dump,jhat内置了一个微型的HTTP/HTML服务器,生成dump的分析结果后,可以在浏览器中查看
  • jstack,用于生成java虚拟机当前时刻的线程快照。
  • jinfo,JVM Configuration info 这个命令作用是实时查看和调整虚拟机运行参数。
18、常见调优工具有哪些(!!!)

常用调优工具分为两类,jdk自带监控工具:jconsole和jvisualvm,第三方有:MAT(Memory Analyzer Tool)、GChisto

  • jconsole,Java Monitoring and Management Console是从java5开始,在JDK中自带的java监控和管理控制台,用于对JVM中内存,线程和类等的监控
  • jvisualvm,jdk自带全能工具,可以分析内存快照、线程快照;监控内存变化、GC变化等。
  • MAT,Memory Analyzer Tool,一个基于Eclipse的内存分析工具,是一个快速、功能丰富的Java heap分析工具,它可以帮助我们查找内存泄漏和减少内存消耗
  • GChisto,一款专业分析gc日志的工具
19、Minor GC与Full GC分别在什么时候发生?

新生代内存不够用时候发生MGC也叫YGC,JVM内存不够的时候发生FGC

20、你知道哪些JVM性能调优参数?!!!

「堆栈内存相关」

  • -Xms 设置初始堆的大小
  • -Xmx设置最大堆的大小
  • -Xmn设置年轻代大小,相当于同时配置-XX:NewSize和-XX:MaxNewSize为一样的值
)
  • -Xss 每个线程的堆栈大小
:
  • -XX:NewSize设置年轻代大小(for 1.3/1.4)
  • -XX:MaxNewSize年轻代最大值(for 1.3/1.4)
  • -XX:NewRatio年轻代与老年代的比值(除去永久代)
  • -XX:SurvivorRatioEden区与Survivor区的的比值
  • -XX:PretenureSizeThreshold当创建的对象超过指定大小时,直接把对象分配在老年代。
  • -XX:MaxTenuringThreshold设定对象在Survivor复制的最大年龄阈值,超过阀值转移到老年代

「垃圾收集器相关」

  • -XX:+UseParallelGC:选择垃圾收集器为并行收集器。
  • -XX:ParallelGCThreads=20:配置并行收集器的线程数
  • -XX:+UseConcMarkSweepGC:设置年老代为并发收集。
  • -XX:CMSFullGCsBeforeCompaction=5 由于并发收集器不对内存空间进行压缩、整理,所以运行一段时间以后会产生"碎片",使得运行效率降低。此值设置运行5次GC以后对内存空间进行压缩、整理。
  • -XX:+UseCMSCompactAtFullCollection:打开对年老代的压缩。可能会影响性能,但是可以消除碎片

「辅助信息相关」

  • -XX:+PrintGCDetails 打印GC详细信息
  • -XX:+HeapDumpOnOutOfMemoryError让JVM在发生内存溢出的时候自动生成内存快照,排查问题用
  • -XX:+DisableExplicitGC禁止系统System.gc(),防止手动误触发FGC造成问题.
  • -XX:+PrintTLAB 查看TLAB空间的使用情况
21、对象一定分配在堆中吗?有没有了解逃逸分析技术?

对象一定分配在堆中吗? 」 不一定的,JVM通过「逃逸分析 」,那些逃不出方法的对象会在栈上分配。

什么是逃逸分析?

逃逸分析,是一种可以有效减少Java对象中同步负载和内存堆分配压力的跨函数全局数据流分析算法。通过逃逸分析,Java HotSPot编译器能够分析出一个新的对象的引用的使用范围,从而决定是否要将这个对象分配到堆上。

逃逸分析是指分析指针动态范围的方法,它同编译器优化原理的指针分析和外形分析相关联。当变量(或者对象)在方法中分配后,其指针有可能被返回或者被全局引用,这样就会被其他方法或者线程所引用,这种现象称作指针(或者引用)的逃逸。通俗点讲,如果一个对象的指针被多个方法或者线程引用时,那么我们就称这个对象的指针发生了逃逸。

逃逸分析的好处:

  • 栈上分配,可以降低垃圾收集器运行的频率;
  • 同步消除,如果发现某个对象只能从一个线程可访问,那么在这个对象上的操作可以不需要同步。
  • 标量替换,把对象分解成一个个基本类型,并且内存分配不再是分配在堆上,而且分配在栈上。这样的好处有,一、减少内存使用,因为不用生成对象头。二、程序内存回收效率高,并且GC频率也会减少。
22、虚拟机为什么使用元空间替换了永久代?(!!!)

「什么是元空间?什么是永久代?为什么用元空间代替永久代?」 我们先回顾一下「方法区」吧:

方法区和堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译后的代码等数据。

「什么是永久代?它和方法区有什么关系呢?」

如果在HotSpot虚拟机上开发、部署,很多程序员都把方法区称作永久代。可以说方法区是规范,永久代是Hotspot针对该规范进行的实现。在Java7及以前的版本,方法区都是永久代实现的。

「什么是元空间?它和方法区有什么关系呢?」

对于Java8,在Hotspot取消了永久代,取而代之的事元空间(Metaspace)。换句话说,就是方法区还是在的,只是实现变了,从永久代变为元空间了。

「为什么使用元空间替换了永久代?」

  • 永久代的方法去,和堆使用的物理内存是连续的。

    永久代」是通过以下这两个参数配置大小的:
  • -XX:PremSIze:设置永久代的初始大小
  • -XX:MaxPermSize:设置永久代的最大值,默认是64M。

对于「永久代 」,如果动态生成很多class的话,就很可能出现「java.lang.OutOfMemoryError:PermGen space错误」,因为永久代空间配置有限嘛。最典型的场景是,在web开发比较多jsp页面的时候。

  • JDK8之后,方法区存在于元空间(Metaspace)。物理内存不再与堆连接,而是直接存在于本地内存中,理论上机器「内存有多大,元空间就有多大」。

可以通过以下的参数来设置元空间的大小:

  • -XX:MetaspaceSize:初始空间大小,达到该值就会触发垃圾收集进行类型卸载,同时GC会对该值进行调整:如果释放了大量的空间,就适当降低该值;如果释放了很少的空间,那么在不超过MaxMataspaceSize,最大空间,默认是没有限制的。
  • -XX:MaxMetaspaceSize:最大空间,默认是没有限制的。
  • -XX:MinMertaspaceFreeRatio:在GC之后,最小的Metaspace剩余空间容量的百分比,减少为分配空间所导致的垃圾收集。
  • -XX:MaxMetaspaceFreeRatio:在GC之后,最大的Metaspace剩余空间容量的百分比,减少为释放空间所导致的垃圾收集,

「所以,为什么使用元空间替换永久代?」

表面上看是为了避免OOM异常。因为通常使用PermSize和MaxPermSize设置永久代的大小就决定了永久代的上限,但是不是总能知道应该设置为多大合适,如果使用默认值很容易遇到OOM错误。当使用元空间时,可以加载多少类的元数据就不再由MaxPermSize控制,而由系统的实际可用空间来控制啦。

23、什么是Stop The World?什么是OopMap?什么是安全点?(!!!!)

进行垃圾回收的过程中,会涉及对象的移动。为了保证对象引用更新的正确性,必须暂停所以后的用户线程,像这样的停顿,虚拟机设计者形象描述为「Stop The World」。也简称为STW。

在HotSpot中,有个数据结构(映射表)称为「OopMap 」。一旦类加载动作完成的时候,HotSpot就会把对象内什么偏移量上是什么类型的数据计算出来,记录到OopMap。在即时编译过程中,也会在「特定的位置 」生成 OopMap,记录下栈上和寄存器里哪些位置是引用。

这些特定的位置主要在:

  1. 循环的末尾(非counted循环)
  2. 方法临返回前/调用方式的call指令后
  3. 可能抛异常的位置

这些位置就叫作「安全点(safepoint)」。用户程序执行时并非在代码指令流的任意位置都能够在停顿下来开始垃圾收集,而是必须是执行到安全点才能够暂停。

24、说一下JVM的主要组成部分及其作用?

JVM包含两个子系统和两个组件,分别为:

  • Class loader(类装载子系统)
  • Execution engine(执行引擎子系统)
  • Runtime data area(运行时数据区组件)
  • Native Interface(本地接口组件)
  • 「Class loader(类装载):」 根据给定的全限定名类名(如:java.lang.Object)来装载class文件到运行时数据区的方法区中。
  • 「Execution engine(执行引擎)」:执行class的指令。
  • 「Native Interface(本地接口)」 :与native lib交互,是其它编程语言交互的接口。
  • 「Runtime data area(运行时数据区域)」:即我们常说的JVM的内存。

首先通过编译器把Java源代码转换为字节码,Class loader(类加载)再把字节码加载到内存中,将其放在运行时数据区的方法区内,而字节码文件只是JVM的一套指令集规范,并不能直接交给底层操作系统去执行,因此需要特定的命令解析气执行引擎,将字节码翻译成底层系统指令,再交由CPU去执行,而这个过程中需要调用其他语言的本地库接口(Native Interface)来实现整个程序的功能。

25、什么是指针碰撞?

一般情况下,JVM的对象都放在堆内存中(发生逃逸分析除外)。当类加载检查通过后,Java虚拟机开始为新生对象分配内存。如果Java堆中内存是绝对规整的,所有被使用过的内存都被放在一边,空间的内存放到另外一边,中间放着一个指针作为分界点的指示器所分配内存仅仅是把那个指针向空闲空间挪动一段与对象大小相等的实例,这种分配方式就是指针碰撞。

26,什么是空闲列表?

如果Java堆内存中的内存并不是规整的,已被使用的内存和空闲的内存相互交错在一起,不可以进行指针碰撞,虚拟机必须维护一个列表,记录哪些内存是可用的,在分配的时候从列表找到一块大的空间分配给对象实例,并更新列表上的记录,这种分配方式就是空闲列表。

27,什么是TLAB?

可以把内存分配的动作按照线程划分在不同的空间之中进行,每个线程在Java堆中预先分配一小块内存,这就是TLAB(Thread Local Allocation Buffer,本地线程分配缓存)。虚拟机通过-XX:UseTLAB设定它的。

30、说一下JVM有哪些垃圾回收器?

如果说垃圾收集算法是内存回收的方法论,那么垃圾收集器就是内存回收的具体实现。

下图展示了7种作用于不同分代的收集器,其中用于回收新生代的收集器包括Serial、PraNew、Parallel Scavenge,回收老年代的收集器包括Serial Old、Parallel Old、CMS,还有用于回收整个Java堆的G1收集器。不同收集器之间的连线表示它们可以搭配使用。

  • Serial收集器(复制算法): 新生代单线程收集器,标记和清理都是单线程,优点是简单高效;

  • ParNew收集器 (复制算法): 新生代收并行集器,实际上是Serial收集器的多线程版本,在多核CPU环境下有着比Serial更好的表现;

  • Parallel Scavenge收集器 (复制算法): 新生代并行收集器,追求高吞吐量,高效利用 CPU。吞吐量 = 用户线程时间/(用户线程时间+GC线程时间),高吞吐量可以高效率的利用CPU时间,尽快完成程序的运算任务,适合后台应用等对交互相应要求不高的场景;

  • Serial Old收集器 (标记-整理算法): 老年代单线程收集器,Serial收集器的老年代版本;

  • Parallel Old收集器 (标记-整理算法): 老年代并行收集器,吞吐量优先,Parallel Scavenge收集器的老年代版本;

  • CMS(Concurrent Mark Sweep)收集器(标记-清除算法): 老年代并行收集器,以获取最短回收停顿时间为目标的收集器,具有高并发、低停顿的特点,追求最短GC回收停顿时间。

  • G1(Garbage First)收集器 (标记-整理算法): Java堆并行收集器,G1收集器是JDK1.7提供的一个新收集器,G1收集器基于"标记-整理"算法实现,也就是说不会产生内存碎片。此外,G1收集器不同于之前的收集器的一个重要特点是:G1回收的范围是整个Java堆(包括新生代,老年代),而前六种收集器回收的范围仅限于新生代或老年代。

  • ZGC (Z Garbage Collector)是一款由Oracle公司研发的,以低延迟为首要目标的一款垃圾收集器。它是基于动态Region内存布局,(暂时)不设年龄分代,使用了读屏障、染色指针和内存多重映射等技术来实现可并发的标记-整理算法的收集器。在 JDK 11 新加入,还在实验阶段,主要特点是:回收TB级内存(最大4T),停顿时间不超过10ms。

    • 优点:低停顿,高吞吐量, ZGC 收集过程中额外耗费的内存小。
    • 缺点:浮动垃圾
      目前使用的非常少,真正普及还是需要些时间的。
  • 新生代收集器:Serial、 ParNew 、 Parallel Scavenge

  • 老年代收集器: CMS 、Serial Old、Parallel Old

  • 整堆收集器: G1 , ZGC (因为不涉年代不在图中)。

31、如何选择垃圾收集器?
  1. 如果你的堆大小不是很大(比如100MB),选择串行收集器一般是效率最高的。
    参数:-XX:+UseSeraiGC
  2. 如果你的应用运行在单核的机器上,或者你的虚拟机核数只有单核,选择串行收集器依然是合适的,这时候启用一些并行收集器没有任何收益。
    参数:-XX:+UseSeraiGC
  3. 如果你的应用是"吞吐量"优先的,并且对较长时间的停顿没有什么特别的要求。选择并行收集器是比较好的。
    参数: -XX:+UseParallelGC
  4. 如果你的应用对响应时间要求较高,想要较少的停顿。甚至 1 秒的停顿都会引起大量的请求失败,那么选择 G1 、 ZGC 、 CMS 都是合理的。虽然这些收集器的 GC 停顿通常都比较短,但它需要一些额外的资源去处理这些工作,通常吞吐量会低一些。
    参数:-XX:+UseConcMarkSweepGC 、-XX:+UseG1GC、-XX:+UseZGC 等。

从上面这些出发点来看,我们平常的Web服务器,都是对响应性要求比较高的。选择性其实就是集中在 CMS 、 G1 、 ZGC 上。而对于某些定时任务,使用并行收集器,是一个比较好的选择。

32、什么是类加载器?

类加载器是一个用来加载类文件的类。Java源代码通过javac编译器编译成类文件。然后JVM来执行类文件中的字节码来执行程序。类加载器负责加载文件系统、网络或其他来源的类文件。

33、什么是tomcat 类加载机制? (!!!)

在tomcat中类的加载稍有不同,如下图:
当tomcat启动时,会创建几种类加载器:Bootstrap引导类加载器 加载JVM启动所需的类,以及标准扩展类(位于jre/lib/ext下)System系统类加载器加载tomcat启动的类,比如bootstrap.jar,通常在catalina.bat或者catalina.sh中指定。位于CATAINA_HOME/bin下。

相关推荐
飛_5 分钟前
解决VSCode无法加载Json架构问题
java·服务器·前端
木棉软糖3 小时前
一个MySQL的数据表最多能够存多少的数据?
java
程序视点3 小时前
Java BigDecimal详解:小数精确计算、使用方法与常见问题解决方案
java·后端
愿你天黑有灯下雨有伞3 小时前
Spring Boot SSE实战:SseEmitter实现多客户端事件广播与心跳保活
java·spring boot·spring
Java初学者小白4 小时前
秋招Day20 - 微服务
java
狐小粟同学4 小时前
JavaEE--3.多线程
java·开发语言·java-ee
KNeeg_5 小时前
Spring循环依赖以及三个级别缓存
java·spring·缓存
AI_Gump6 小时前
【AI阅读】20250717阅读输入
java·spring boot·spring
找不到、了6 小时前
Java排序算法之<插入排序>
java·算法·排序算法
设计师小聂!7 小时前
力扣热题100----------53最大子数组和
java·数据结构·算法·leetcode