文章目录
- 1. 本地方法栈
- 2. 堆
- 3. 方法区
1. 本地方法栈
本地方法栈和虚拟机栈有点类似,均具有线程隔离的特点以及都能抛出StackOverflowError和OutOfMemoryError异常。
但是不同之处在于本地方法栈服务的对象是JVM执行的native方法,而虚拟机栈服务的是JVM执行的Java方法。
带有native关键词的方法就是需要Java去调用本地的C或C++方法,但是Java有时候无法直接和操作系统底层交互,所以就用到了本地方法栈。
2. 堆
通过new出来的对象都会放在堆里
堆有什么特点?
- 它是线程共享,堆内存中的对象都需要考虑线程安全问题
- 有垃圾回收机制
堆内存溢出
java.lang.OutofMemoryError :java heap space. 堆内存溢出
可以使用 -Xmx8m 来指定堆内存大小。
堆内存诊断
- jps 工具
查看当前系统中有哪些 java 进程 - jmap 工具
查看堆内存占用情况 jmap - heap 进程id - jconsole 工具
图形界面的,多功能的监测工具,可以连续监测 - jvisualvm 工具
3. 方法区
在JDK以前,习惯上把方法区称为永生代,但是到了JDK8,完全废弃了永生代的概念,改用在本地内存中实现的元空间。
这两个最大的区别就是:元空间不在虚拟机设置的内存中,而是使用本地内存
方法区它是用于存储一些已经被虚拟机加载的类型信息、常量、静态变量、即时编译器编译后的代码缓存,运行常量池等
方法区的内存溢出
- 1.8 之前会导致永久代内存溢出
- 使用 -XX:MaxPermSize=8m 指定永久代内存大小
- 1.8 之后会导致元空间内存溢出
- 使用 -XX:MaxMetaspaceSize=8m 指定元空间大小
运行时常量池
二进制字节码包含(类的基本信息,常量池,类方法定义,包含了虚拟机的指令)
首先看看常量池是什么,编译如下代码:
public class Test {
public static void main(String[] args) {
System.out.println("Hello World!");
}
}
每条指令都会对应常量池表中一个地址,常量池表中的地址可能对应着一个类名、方法名、参数类型等信息。
常量池:
就是一张表,虚拟机指令根据这张常量表找到要执行的类名、方法名、参数类型、字面量信息
运行时常量池:
常量池是 *.class 文件中的,当该类被加载以后,它的常量池信息就会放入运行时常量池,并把里面的符号地址变为真实地址
参考:JVM 学习笔记(一)内存结构_CodeAli的博客-CSDN博客