目录
JVM组成部分图解
内存结构
JVM内存结构主要有五大部分组成
- 程序计数器
- 栈
- 本地方法栈
- 堆
- 方法区
程序计数器
作用
简单来说, 是记住下一条jvm指令的执行地址
特点
- 线程私有
- 唯一不会出现内存溢出的区域
虚拟机栈
- 每个线程运行时所需的内存成为虚拟机栈
- 每个栈由多个栈帧组成, 栈帧对应每个方法被调用时所占用的内存
- 每个线程只能有一个活动栈帧, 对应着当前正在执行的那个方法
- 存放一些方法的参数, 局部变量, 返回地址等等
- 线程私有
问题辨析
- 垃圾回收是否涉及到栈内存?
不设计, 因为栈只是存放方法被调用时的参数, 局部变量, 用完之后自动被弹出栈- 栈内存分配越大越好吗
不是, 内存越大, 线程数会变少, 物理内存是不变的, 分配的越多, 反而线程数会变少- 方法内的局部变量是否是线程安全
是的, 如果多个线程调用相同方法时, 都会各自分配一个栈内存, 所以局部变量是私有的
栈内存溢出 (StackOverFlowError)
- 栈帧过多导致栈内存溢出 (递归调用没有正确的设置停止语句)
- 栈帧过大
- 设置栈内存大小 -Xss256k
本地方法栈
为本地方法代码执行时所占用的内存空间, jdk方法被native修饰的方法
堆
通过new关键字, 创建的对象都会使用堆内存
- 线程共享, 堆中对象都需要考虑线程安全问题
- 有垃圾回收机制
堆内存溢出
当对象没有被回收掉(对象又被调用), 对象占满了堆, 就会内存溢出
参数设置
-Xmx1204m
相关的内存诊断工具
- jps工具
查看当前系统中有哪些java进程- jmap工具
查看堆内存占用情况 jmap -heap 进程id- jconsole工具
图像化界面, 多功能的监测工具, 可以连续监测
方法区
- 方法区是被所有线程共享,所有字段和方法字节码,以及一些特殊的方法,如构造函数,接口代码也在此定义,简单说,所有定义的方法信息都保存在该区域。静态变量、静态方法、常量、类信息(构造方法、接口定义)、运行时的常量池存在方法区中,但是实例存在堆内存中,和方法区无关。
- 线程共享