目录
1. 程序计数器
1.1 定义
JVM中的程序计数寄存器(Program Counter Register)中Register的命名来源于CPU的寄存器,寄存器用来存储指令相关的现场信息。
CPU只有把数据装载到寄存器中才能够正常执行指令。
JVM中的寄存器不是物理上的寄存器而是对物理寄存器的抽象模拟。
程序计数器有时候也被称为为程序钩子
在JVM规范中,每个线程都有它自己的程序计数器,是线程私有的,生命周期与线程的生命周期保持一致。
- 作用
- 是记住下一条jvm指令的执行地址
- 是记住下一条jvm指令的执行地址
- 特点
- 是线程私有的
- 不会存在内存溢出( 为什么JVM规范中,程序计数器不会发生内存溢出)
2. 虚拟机栈
2.1 定义
Java Virtual Machine Stacks (Java 虚拟机栈)
-
每个线程运行时所需要的内存,称为虚拟机栈
-
每个栈由多个栈帧(Frame)组成,对应着每次方法调用时所占用的内存
-
每个线程只能有一个活动栈帧,对应着当前正在执行的那个方法
问题辨析
- 垃圾回收是否涉及栈内存?
不需要,栈内存是方法调用时产生的栈帧内存,会在方法结束后弹出栈被自动回收
- 栈内存分配越大越好吗?
栈内存可以通过虚拟机参数指定 -Xss
栈内存越大,线程数越少
假设500M的物理内存,每个栈内存1M那么可以有500个线程
如果栈内存设置为2M那么只能有250个线程
一般使用默认栈内存
- 方法内的局部变量是否线程安全?
- 如果方法内局部变量没有逃离方法的作用访问,它是线程安全的
- 如果是局部变量引用了对象,并逃离方法的作用范围 ,需要考虑线程安全
2.2 栈内存溢出
- 栈帧过多导致栈内存溢出(方法递归调用)
- 栈帧过大导致栈内存溢出
2.3 线程运行诊断
案例1: cpu 占用过多
定位
- 用top定位哪个进程对cpu的占用过高
- ps H -eo pid,tid,%cpu | grep 进程id (用ps命令进一步定位是哪个线程引起的cpu占用过高)
- jstack 进程id
- 可以根据线程id 找到有问题的线程,进一步定位到问题代码的源码行号
案例2:程序运行很长时间没有结果
有可能是由于死锁问题导致。
jstack 查看死锁
3. 本地方法栈
Java虚拟机调用 本地方法 时,需要给本地方法提供的内存空间。
以上3种内存结构都是线程私有的
4. 堆
4.1 定义
Heap 堆
- 通过 new 关键字,创建对象都会使用堆内存
特点
- 它是线程共享的,堆中对象都需要考虑线程安全的问题
- 有垃圾回收机制
4.2 堆内存溢出
定义:申请内存时候 没有足够得内存使用就是内存溢出;
原因:对象没有使用才会被回收,如果对象较多,且都在使用则会产生内存溢出
-Xmx可以设置堆空间大小(可以快速排查内存溢出)
内存泄漏:对象没有被用到,但是又无法被GC回收就是内存泄露;
分类:
- 经常发生: 发生内存泄露的代码会被多次执行,每次执行,泄露一块内存
- 偶然发生: 在某些特定情况下才会发生
- 一次性: 发生内存泄露的方法只会执行一次
- 隐式泄漏: 一直占着内存不释放,直到执行结束; 严格的说这个不算内存泄漏,因为最终释放掉了, 但是如果执行时间特别长,也可能会导致内存耗尽
4.3 堆内存诊断
- jps 工具
查看当前系统中有哪些 java 进程 - jmap 工具
查看堆内存占用情况 jmap - heap 进程id - jconsole 工具
图形界面的,多功能的监测工具,可以连续监测
案例
- 垃圾回收后,内存占用仍然很高
jdk8使用JVisualVM
jdk9使用jmc