JVM内存区域
程序计数器
可以看做当前线程执行的字节码的行号指示器,可以记录正在执行虚拟机字节码的指令的地址(如果执行的是本地方法则为空)
特点
- 线程私有
- 不会内存溢出
虚拟机栈
定义: 每个线程运行时需要的内存,成为虚拟机栈
每个栈由多个栈帧组成,对应着每次方法调用时所占用的内存
每个线程只能有一个活动栈(栈顶部的方法),对应着当当前正在执行的那个方法
栈的结构: 先进后出
栈: 线程运行需要的内存空间
栈帧: 每个方法运行时需要的内存
线程私有
问题
-
垃圾回收会不会涉及栈内存? 不会
-
栈内存越大越好吗? 栈内存越大, 线程越小
-
方法内的局部变量是不是线程安全? 局部变量是私有的,线程安全的, 但是加了static是需要考虑线程安全
栈内存溢出
当线程请求的栈深度超过最大值,会抛出 StackOverflowError 异常;
栈进行动态扩展时如果无法申请到足够内存,会抛出 OutOfMemoryError 异常。
-Xss256k 设置栈内存大小
- 栈帧过多导致溢出, eg : 方法的递归,没有写终止条件
- 栈帧过大
本地方法栈
Native Method Stacks
本地方法栈与 Java 虚拟机栈类似,它们之间的区别只不过是本地方法栈为本地方法服务。
本地方法一般是用其它语言(C、C++ 或汇编语言等)编写的,并且被编译为基于本机硬件和操作系统的程序,对待这些方法需要特别处理。
堆(Heap)
- 通过new关键字,创建的对象都会使用堆内存 , 用于存放对象的实例
特点
- 线程共享,需要考虑线程安全
- 有垃圾回收机制
堆内存诊断
- jps工具 : 查看当前系统有哪些java 进程
- jmap工具 : 查看堆内存占用情况
- jconsole工具 : 图形界面
方法区
用于存放已被加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。线程共享
方法区是一个 JVM 规范,永久代与元空间都是其一种实现方式。在 JDK 1.8 之后,原来永久代的数据被分到了堆和元空间中。元空间存储类的元信息,静态变量和常量池等放入堆中。
运行时常量池
常量池:就是一个张表, 虚拟机指令根据这张常量表找到要执行的类名,方法名,参数类型,字面量等信息
运行时常量池是方法区的一部分。
Class 文件中的常量池(编译器生成的字面量和符号引用)会在类加载后被放入这个区域。
StringTable 串池 待学习
直接内存
直接内存不是JVM运行时数据区的一部分
- 常见于NIO(NEW Input/Ouput)操作时,用于数据缓冲区
- 分配回收成本高,但读写性能高
- 不受JVM内存回收管理
参考
《深入理解java虚拟机》