智能摘要文章源自JAVA秀-https://www.javaxiu.com/42040.html
内存是非常重要的系统资源,是硬盘和CPU的中间仓库及桥梁,承载着操作系统和应用程序的实时运行。它是内存区域中最大的一块区域,被所有线程共享,存储着几乎所有的实例对象、数组。简而言之,JVM代码缓存是JVM将其字节码存储为本机代码的区域。这就是为什么一些开发人员将此内存称为JIT代码缓存的原因。提供近3W行代码的SpringBoot示例,以及超4W行代码的电商微服务项目。文章源自JAVA秀-https://www.javaxiu.com/42040.html
原文约 8813 字 | 图片 15 张 | 建议阅读 18 分钟 | 评价反馈文章源自JAVA秀-https://www.javaxiu.com/42040.html
精美图文带你掌握 JVM 内存布局
点击关注 ? 芋道源码 文章源自JAVA秀-https://www.javaxiu.com/42040.html
收录于话题文章源自JAVA秀-https://www.javaxiu.com/42040.html
#芋道源码254文章源自JAVA秀-https://www.javaxiu.com/42040.html
#成为架构师1文章源自JAVA秀-https://www.javaxiu.com/42040.html
点击上方“芋道源码”,选择“设为星标”文章源自JAVA秀-https://www.javaxiu.com/42040.html
管她前浪,还是后浪?文章源自JAVA秀-https://www.javaxiu.com/42040.html
能浪的浪,才是好浪!文章源自JAVA秀-https://www.javaxiu.com/42040.html
每天 10:33 更新文章,每天掉亿点点头发...文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html 源码精品专栏文章源自JAVA秀-https://www.javaxiu.com/42040.html
原创 | Java 2021 超神之路,很肝~文章源自JAVA秀-https://www.javaxiu.com/42040.html
中文详细注释的开源项目文章源自JAVA秀-https://www.javaxiu.com/42040.html
RPC 框架 Dubbo 源码解析文章源自JAVA秀-https://www.javaxiu.com/42040.html
网络应用框架 Netty 源码解析文章源自JAVA秀-https://www.javaxiu.com/42040.html
消息中间件 RocketMQ 源码解析文章源自JAVA秀-https://www.javaxiu.com/42040.html
数据库中间件 Sharding-JDBC 和 MyCAT 源码解析文章源自JAVA秀-https://www.javaxiu.com/42040.html
作业调度中间件 Elastic-Job 源码解析文章源自JAVA秀-https://www.javaxiu.com/42040.html
分布式事务中间件 TCC-Transaction 源码解析文章源自JAVA秀-https://www.javaxiu.com/42040.html
Eureka 和 Hystrix 源码解析文章源自JAVA秀-https://www.javaxiu.com/42040.html
Java 并发源码文章源自JAVA秀-https://www.javaxiu.com/42040.html
来源:juejin.im/post/6844904033396719624文章源自JAVA秀-https://www.javaxiu.com/42040.html
前言文章源自JAVA秀-https://www.javaxiu.com/42040.html
概览文章源自JAVA秀-https://www.javaxiu.com/42040.html
Heap (堆区)文章源自JAVA秀-https://www.javaxiu.com/42040.html
创建一个新对象内存分配流程文章源自JAVA秀-https://www.javaxiu.com/42040.html
Metaspace 元空间文章源自JAVA秀-https://www.javaxiu.com/42040.html
Java 虚拟机栈文章源自JAVA秀-https://www.javaxiu.com/42040.html
本地方法栈文章源自JAVA秀-https://www.javaxiu.com/42040.html
程序计数器文章源自JAVA秀-https://www.javaxiu.com/42040.html
直接内存文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
前言
本 JVM 系列属于本人学习过程当中总结的一些知识点,目的是想让读者更快地掌握 JVM 相关的知识要点,难免会有所侧重,若想要更加系统更加详细的学习 JVM 知识,还是需要去阅读专业的书籍和文档。文章源自JAVA秀-https://www.javaxiu.com/42040.html
本文主题内容:文章源自JAVA秀-https://www.javaxiu.com/42040.html
JVM 内存区域概览文章源自JAVA秀-https://www.javaxiu.com/42040.html
堆区的空间分配是怎么样?堆溢出的演示文章源自JAVA秀-https://www.javaxiu.com/42040.html
创建一个新对象内存是怎么分配的?文章源自JAVA秀-https://www.javaxiu.com/42040.html
方法区 到 Metaspace 元空间文章源自JAVA秀-https://www.javaxiu.com/42040.html
栈帧是什么?栈帧里有什么?怎么理解?文章源自JAVA秀-https://www.javaxiu.com/42040.html
本地方法栈文章源自JAVA秀-https://www.javaxiu.com/42040.html
程序计数器文章源自JAVA秀-https://www.javaxiu.com/42040.html
Code Cache 是什么?文章源自JAVA秀-https://www.javaxiu.com/42040.html
注:请区分 JVM 内存结构(内存布局)和 JMM(Java 内存模型)这两个不同的概念!文章源自JAVA秀-https://www.javaxiu.com/42040.html
推荐下自己做的 Spring Boot 的实战项目:文章源自JAVA秀-https://www.javaxiu.com/42040.html
https://github.com/YunaiV/ruoyi-vue-pro文章源自JAVA秀-https://www.javaxiu.com/42040.html
概览
内存是非常重要的系统资源,是硬盘和 CPU 的中间仓库及桥梁,承载着操作系统和应用程序的实时运行。JVM 内存布局规定了 Java 在运行过程中内存申请、分配、管理的策略,保证了 JVM 的高效稳定运行。文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
图片文章源自JAVA秀-https://www.javaxiu.com/42040.html
上图描述了当前比较经典的 JVM 内存布局。(堆区画小了 2333,按理来说应该是最大的区域)文章源自JAVA秀-https://www.javaxiu.com/42040.html
如果按照线程是否共享来分类的话,如下图所示:文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
图片文章源自JAVA秀-https://www.javaxiu.com/42040.html
PS:线程是否共享这点,实际上理解了每块区域的实际用处之后,就很自然而然的就记住了。不需要死记硬背。文章源自JAVA秀-https://www.javaxiu.com/42040.html
下面让我们来了解下各个区域。文章源自JAVA秀-https://www.javaxiu.com/42040.html
推荐下自己做的 Spring Cloud 的实战项目:文章源自JAVA秀-https://www.javaxiu.com/42040.html
https://github.com/YunaiV/onemall文章源自JAVA秀-https://www.javaxiu.com/42040.html
Heap (堆区)
1. 堆区的介绍
我们先来说堆。堆是 OOM 故障最主要的发生区域。它是内存区域中最大的一块区域,被所有线程共享,存储着几乎所有的实例对象、数组。所有的对象实例以及数组都要在堆上分配,但是随着 JIT 编译器的发展与逃逸分析技术逐渐成熟,栈上分配、标量替换优化技术将会导致一些微妙的变化发生,所有的对象都分配在堆上也渐渐变得不是那么“绝对”了。文章源自JAVA秀-https://www.javaxiu.com/42040.html
Java 堆是垃圾收集器管理的主要区域,因此很多时候也被称做“GC 堆”。从内存回收的角度来看,由于现在收集器基本都采用分代收集算法,所以 Java 堆中还可以细分为:新生代和老年代。再细致一点的有 Eden 空间、From Survivor 空间、To Survivor 空间等。从内存分配的角度来看,线程共享的 Java 堆中可能划分出多个线程私有的分配缓冲区(Thread Local Allocation Buffer,TLAB)。不过无论如何划分,都与存放内容无关,无论哪个区域,存储的都仍然是对象实例,进一步划分的目的是为了更好地回收内存,或者更快地分配内存。文章源自JAVA秀-https://www.javaxiu.com/42040.html
2. 堆区的调整
根据 Java 虚拟机规范的规定,Java 堆可以处于物理上不连续的内存空间中,只要逻辑上是连续的即可,就像我们的磁盘空间一样。在实现时,既可以实现成固定大小的,也可以在运行时动态地调整。文章源自JAVA秀-https://www.javaxiu.com/42040.html
如何调整呢?文章源自JAVA秀-https://www.javaxiu.com/42040.html
通过设置如下参数,可以设定堆区的初始值和最大值,比如 -Xms256M -Xmx 1024M,其中 -X 这个字母代表它是 JVM 运行时参数,ms 是 memory start 的简称,中文意思就是内存初始值,mx 是 memory max 的简称,意思就是最大内存。文章源自JAVA秀-https://www.javaxiu.com/42040.html
值得注意的是,在通常情况下,服务器在运行过程中,堆空间不断地扩容与回缩,会形成不必要的系统压力所以在线上生产环境中 JVM 的 Xms 和 Xmx 会设置成同样大小,避免在 GC 后调整堆大小时带来的额外压力。文章源自JAVA秀-https://www.javaxiu.com/42040.html
3. 堆的默认空间分配
另外,再强调一下堆空间内存分配的大体情况。文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
图片文章源自JAVA秀-https://www.javaxiu.com/42040.html
这里可能就会有人来问了,你从哪里知道的呢?如果我想配置这个比例,要怎么修改呢?文章源自JAVA秀-https://www.javaxiu.com/42040.html
我先来告诉你怎么看虚拟机的默认配置。命令行上执行如下命令,就可以查看当前 JDK 版本所有默认的 JVM 参数。文章源自JAVA秀-https://www.javaxiu.com/42040.html
java -XX:+PrintFlagsFinal -version
输出
对应的输出应该有几百行,我们这里去看和堆内存分配相关的两个参数文章源自JAVA秀-https://www.javaxiu.com/42040.html
>java -XX:+PrintFlagsFinal -version[Global flags] ... uintx InitialSurvivorRatio = 8 uintx NewRatio = 2 ...java version "1.8.0_131"Java(TM) SE Runtime Environment (build 1.8.0_131-b11)Java HotSpot(TM) 64-Bit Server VM (build 25.131-b11, mixed mode)
参数解释
文章源自JAVA秀-https://www.javaxiu.com/42040.html
图片文章源自JAVA秀-https://www.javaxiu.com/42040.html
因为新生代是由 Eden + S0 + S1 组成的,所以按照上述默认比例,如果 eden 区内存大小是 40M,那么两个 survivor 区就是 5M,整个 young 区就是 50M,然后可以算出 Old 区内存大小是 100M,堆区总大小就是 150M。文章源自JAVA秀-https://www.javaxiu.com/42040.html
4. 堆溢出演示
/** * VM Args:-Xms10m -Xmx10m -XX:+HeapDumpOnOutOfMemoryError * @author Richard_Yi */public class HeapOOMTest { public static final int _1MB = 1024 * 1024; public static void main(String[] args) { List<byte[]> byteList = new ArrayList<>(10); for (int i = 0; i < 10; i++) { byte[] bytes = new byte[2 * _1MB]; byteList.add(bytes); } }}
输出
java.lang.OutOfMemoryError: Java heap spaceDumping heap to java_pid32372.hprof ...Heap dump file created [7774077 bytes in 0.009 secs]Exception in thread "main" java.lang.OutOfMemoryError: Java heap space at jvm.HeapOOMTest.main(HeapOOMTest.java:18)
-XX:+HeapDumpOnOutOfMemoryError 可以让 JVM 在遇到 OOM 异常时,输出堆内信息,特别是对相隔数月才出现的 OOM 异常尤为重要。文章源自JAVA秀-https://www.javaxiu.com/42040.html
创建一个新对象内存分配流程
看完上面对堆的介绍,我们趁热打铁再学习一下 JVM 创建一个新对象的内存分配流程。文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
图片文章源自JAVA秀-https://www.javaxiu.com/42040.html
绝大部分对象在 Eden 区生成,当 Eden 区装填满的时候,会触发 Young Garbage Collection,即 YGC。垃圾回收的时候,在 Eden 区实现清除策略,没有被引用的对象则直接回收。依然存活的对象会被移送到 Survivor 区。Survivor 区分为 so 和 s1 两块内存空间。每次 YGC 的时候,它们将存活的对象复制到未使用的那块空间,然后将当前正在使用的空间完全清除,交换两块空间的使用状态。如果 YGC 要移送的对象大于 Survivor 区容量的上限,则直接移交给老年代。一个对象也不可能永远呆在新生代,就像人到了 18 岁就会成年一样,在 JVM 中 -XX:MaxTenuringThreshold 参数就是来配置一个对象从新生代晋升到老年代的阈值。默认值是 15,可以在 Survivor 区交换 14 次之后,晋升至老年代。文章源自JAVA秀-https://www.javaxiu.com/42040.html
上述涉及到一部分垃圾回收的名词,不熟悉的读者可以查阅资料或者看下本系列的垃圾回收章节。文章源自JAVA秀-https://www.javaxiu.com/42040.html
Metaspace 元空间
在 HotSpot JVM 中,永久代( ≈ 方法区)中用于存放类和方法的元数据以及常量池,比如 Class 和 Method。每当一个类初次被加载的时候,它的元数据都会放到永久代中。文章源自JAVA秀-https://www.javaxiu.com/42040.html
永久代是有大小限制的,因此如果加载的类太多,很有可能导致永久代内存溢出,即万恶的 java.lang.OutOfMemoryError: PermGen,为此我们不得不对虚拟机做调优。文章源自JAVA秀-https://www.javaxiu.com/42040.html
那么,Java 8 中 PermGen 为什么被移出 HotSpot JVM 了?(详见:JEP 122: Remove the Permanent Generation):文章源自JAVA秀-https://www.javaxiu.com/42040.html
由于 PermGen 内存经常会溢出,引发恼人的 java.lang.OutOfMemoryError: PermGen,因此 JVM 的开发者希望这一块内存可以更灵活地被管理,不要再经常出现这样的 OOM文章源自JAVA秀-https://www.javaxiu.com/42040.html
移除 PermGen 可以促进 HotSpot JVM 与 JRockit VM 的融合,因为 JRockit 没有永久代。文章源自JAVA秀-https://www.javaxiu.com/42040.html
根据上面的各种原因,PermGen 最终被移除,方法区移至 Metaspace,字符串常量池移至堆区。文章源自JAVA秀-https://www.javaxiu.com/42040.html
准确来说,Perm 区中的字符串常量池被移到了堆内存中是在 Java7 之后,Java 8 时,PermGen 被元空间代替,其他内容比如类元信息、字段、静态属性、方法、常量等都移动到元空间区。比如 java/lang/Object 类元信息、静态属性 System.out、整形常量 100000 等。文章源自JAVA秀-https://www.javaxiu.com/42040.html
元空间的本质和永久代类似,都是对 JVM 规范中方法区的实现。不过元空间与永久代之间最大的区别在于:元空间并不在虚拟机中,而是使用本地内存。因此,默认情况下,元空间的大小仅受本地内存限制。(和后面提到的直接内存一样,都是使用本地内存)文章源自JAVA秀-https://www.javaxiu.com/42040.html
In JDK 8, classes metadata is now stored in the native heap and this space is called Metaspace.文章源自JAVA秀-https://www.javaxiu.com/42040.html
对应的 JVM 调参:文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
图片文章源自JAVA秀-https://www.javaxiu.com/42040.html
Java 虚拟机栈
对于每一个线程,JVM 都会在线程被创建的时候,创建一个单独的栈。也就是说虚拟机栈的生命周期和线程是一致,并且是线程私有的。除了 Native 方法以外,Java 方法都是通过 Java 虚拟机栈来实现调用和执行过程的(需要程序技术器、堆、元空间内数据的配合)。所以 Java 虚拟机栈是虚拟机执行引擎的核心之一。而 Java 虚拟机栈中出栈入栈的元素就称为「栈帧」。文章源自JAVA秀-https://www.javaxiu.com/42040.html
栈帧(Stack Frame)是用于支持虚拟机进行方法调用和方法执行的数据结构。栈帧存储了方法的局部变量表、操作数栈、动态连接和方法返回地址等信息。每一个方法从调用至执行完成的过程,都对应着一个栈帧在虚拟机栈里从入栈到出栈的过程。文章源自JAVA秀-https://www.javaxiu.com/42040.html
栈对应线程,栈帧对应方法文章源自JAVA秀-https://www.javaxiu.com/42040.html
在活动线程中, 只有位于栈顶的帧才是有效的, 称为当前栈帧。正在执行的方法称为当前方法。在执行引擎运行时, 所有指令都只能针对当前栈帧进行操作。而 StackOverflowError 表示请求的栈溢出, 导致内存耗尽, 通常出现在递归方法中。文章源自JAVA秀-https://www.javaxiu.com/42040.html
虚拟机栈通过 pop 和 push 的方式,对每个方法对应的活动栈帧进行运算处理,方法正常执行结束,肯定会跳转到另一个栈帧上。在执行的过程中,如果出现了异常,会进行异常回溯,返回地址通过异常处理表确定。文章源自JAVA秀-https://www.javaxiu.com/42040.html
可以看出栈帧在整个 JVM 体系中的地位颇高。下面也具体介绍一下栈帧中的存储信息。文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
图片文章源自JAVA秀-https://www.javaxiu.com/42040.html
1. 局部变量表
局部变量表就是存放方法参数和方法内部定义的局部变量的区域。文章源自JAVA秀-https://www.javaxiu.com/42040.html
局部变量表所需的内存空间在编译期间完成分配,当进入一个方法时,这个方法需要在帧中分配多大的局部变量空间是完全确定的,在方法运行期间不会改变局部变量表的大小。文章源自JAVA秀-https://www.javaxiu.com/42040.html
这里直接上代码,更好理解。文章源自JAVA秀-https://www.javaxiu.com/42040.html
public int test(int a, int b) { Object obj = new Object(); return a + b;}
如果局部变量是 Java 的 8 种基本基本数据类型,则存在局部变量表中,如果是引用类型。如 new 出来的 String,局部变量表中存的是引用,而实例在堆中。文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
图片文章源自JAVA秀-https://www.javaxiu.com/42040.html
2. 操作栈
操作数栈(Operand Stack)看名字可以知道是一个栈结构。Java 虚拟机的解释执行引擎称为“基于栈的执行引擎”,其中所指的“栈”就是操作数栈。当 JVM 为方法创建栈帧的时候,在栈帧中为方法创建一个操作数栈,保证方法内指令可以完成工作。文章源自JAVA秀-https://www.javaxiu.com/42040.html
还是用实操理解一下。文章源自JAVA秀-https://www.javaxiu.com/42040.html
/** * @author Richard_yyf */public class OperandStackTest { public int sum(int a, int b) { return a + b; }}
编译生成 .class 文件之后,再反汇编查看汇编指令文章源自JAVA秀-https://www.javaxiu.com/42040.html
> javac OperandStackTest.java> javap -v OperandStackTest.class > 1.txt
public int sum(int, int); descriptor: (II)I flags: ACC_PUBLIC Code: stack=2, locals=3, args_size=3 // 最大栈深度为2 局部变量个数为3 0: iload_1 // 局部变量1 压栈 1: iload_2 // 局部变量2 压栈 2: iadd // 栈顶两个元素相加,计算结果压栈 3: ireturn LineNumberTable: line 10: 0
3. 动态连接
每个栈帧中包含一个在常量池中对当前方法的引用, 目的是支持方法调用过程的动态连接。文章源自JAVA秀-https://www.javaxiu.com/42040.html
4. 方法返回地址
方法执行时有两种退出情况:文章源自JAVA秀-https://www.javaxiu.com/42040.html
正常退出,即正常执行到任何方法的返回字节码指令,如 RETURN、IRETURN、ARETURN 等文章源自JAVA秀-https://www.javaxiu.com/42040.html
异常退出文章源自JAVA秀-https://www.javaxiu.com/42040.html
无论何种退出情况,都将返回至方法当前被调用的位置。方法退出的过程相当于弹出当前栈帧,退出可能有三种方式:文章源自JAVA秀-https://www.javaxiu.com/42040.html
返回值压入上层调用栈帧文章源自JAVA秀-https://www.javaxiu.com/42040.html
异常信息抛给能够处理的栈帧文章源自JAVA秀-https://www.javaxiu.com/42040.html
PC 计数器指向方法调用后的下一条指令文章源自JAVA秀-https://www.javaxiu.com/42040.html
本地方法栈
本地方法栈(Native Method Stack)与虚拟机栈所发挥的作用是非常相似的,它们之间的区别不过是虚拟机栈为虚拟机执行 Java 方法(也就是字节码)服务,而本地方法栈则为虚拟机使用到的 Native 方法服务。在虚拟机规范中对本地方法栈中方法使用的语言、使用方式与数据结构并没有强制规定,因此具体的虚拟机可以自由实现它。甚至有的虚拟机(譬如 Sun HotSpot 虚拟机)直接就把本地方法栈和虚拟机栈合二为一。与虚拟机栈一样,本地方法栈区域也会抛出 StackOverflowError 和 OutOfMemoryError 异常。文章源自JAVA秀-https://www.javaxiu.com/42040.html
程序计数器
程序计数器(Program Counter Register)是一块较小的内存空间。是线程私有的。它可以看作是当前线程所执行的字节码的行号指示器。什么意思呢?文章源自JAVA秀-https://www.javaxiu.com/42040.html
白话版本:因为代码是在线程中运行的,线程有可能被挂起。即 CPU 一会执行线程 A,线程 A 还没有执行完被挂起了,接着执行线程 B,最后又来执行线程 A 了,CPU 得知道执行线程A的哪一部分指令,线程计数器会告诉 CPU。文章源自JAVA秀-https://www.javaxiu.com/42040.html
由于 Java 虚拟机的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的,CPU 只有把数据装载到寄存器才能够运行。寄存器存储指令相关的现场信息,由于 CPU 时间片轮限制,众多线程在并发执行过程中,任何一个确定的时刻,一个处理器或者多核处理器中的一个内核,只会执行某个线程中的一条指令。文章源自JAVA秀-https://www.javaxiu.com/42040.html
因此,为了线程切换后能恢复到正确的执行位置,每条线程都需要有一个独立的程序计数器,各条线程之间计数器互不影响,独立存储。每个线程在创建后,都会产生自己的程序计数器和栈帧,程序计数器用来存放执行指令的偏移量和行号指示器等,线程执行或恢复都要依赖程序计数器。此区域也不会发生内存溢出异常。文章源自JAVA秀-https://www.javaxiu.com/42040.html
直接内存
直接内存(Direct Memory)并不是虚拟机运行时数据区的一部分,也不是 Java 虚拟机规范中定义的内存区域。但是这部分内存也被频繁地使用,而且也可能导致 OutOfMemoryError 异常出现,所以我们放到这里一起讲解。文章源自JAVA秀-https://www.javaxiu.com/42040.html
在 JDK 1.4 中新加入了 NIO(New Input/Output)类,引入了一种基于通道(Channel)与缓冲区(Buffer)的 I/O 方式,它可以使用 Native 函数库直接分配堆外内存,然后通过一个存储在 Java 堆中的 DirectByteBuffer 对象作为这块内存的引用进行操作。这样能在一些场景中显著提高性能,因为避免了在 Java 堆和 Native 堆中来回复制数据。文章源自JAVA秀-https://www.javaxiu.com/42040.html
显然,本机直接内存的分配不会受到 Java 堆大小的限制,但是,既然是内存,肯定还是会受到本机总内存(包括 RAM 以及 SWAP 区或者分页文件)大小以及处理器寻址空间的限制。如果内存区域总和大于物理内存的限制,也会出现 OOM。文章源自JAVA秀-https://www.javaxiu.com/42040.html
Code Cache
简而言之, JVM 代码缓存是 JVM 将其字节码存储为本机代码的区域 。我们将可执行本机代码的每个块称为 nmethod。该 nmethod 可能是一个完整的或内联 Java 方法。文章源自JAVA秀-https://www.javaxiu.com/42040.html
实时(JIT)编译器是代码缓存区域的最大消费者。这就是为什么一些开发人员将此内存称为 JIT 代码缓存的原因。文章源自JAVA秀-https://www.javaxiu.com/42040.html
这部分代码所占用的内存空间成为 CodeCache 区域。一般情况下我们是不会关心这部分区域的且大部分开发人员对这块区域也不熟悉。如果这块区域 OOM 了,在日志里面就会看到:文章源自JAVA秀-https://www.javaxiu.com/42040.html
java.lang.OutOfMemoryError code cache。
诊断选项
文章源自JAVA秀-https://www.javaxiu.com/42040.html
图片文章源自JAVA秀-https://www.javaxiu.com/42040.html
- END -文章源自JAVA秀-https://www.javaxiu.com/42040.html
欢迎加入我的知识星球,一起探讨架构,交流源码。加入方式,长按下方二维码噢:文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
已在知识星球更新源码解析如下:文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
最近更新《芋道 SpringBoot 2.X 入门》系列,已经 101 余篇,覆盖了 MyBatis、Redis、MongoDB、ES、分库分表、读写分离、SpringMVC、Webflux、权限、WebSocket、Dubbo、RabbitMQ、RocketMQ、Kafka、性能测试等等内容。文章源自JAVA秀-https://www.javaxiu.com/42040.html
提供近 3W 行代码的 SpringBoot 示例,以及超 4W 行代码的电商微服务项目。文章源自JAVA秀-https://www.javaxiu.com/42040.html
获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章源自JAVA秀-https://www.javaxiu.com/42040.html
文章有帮助的话,在看,转发吧。谢谢支持哟 (*^__^*)文章源自JAVA秀-https://www.javaxiu.com/42040.html
阅读原文文章源自JAVA秀-https://www.javaxiu.com/42040.html

评论