二.JVM内存结构(1)

1. 程序计数器

程序计数器

1.1 定义

JVM中的程序计数寄存器(Program Counter Register)中Register的命名来源于CPU的寄存器,寄存器用来存储指令相关的现场信息。

CPU只有把数据装载到寄存器中才能够正常执行指令。

JVM中的寄存器不是物理上的寄存器而是对物理寄存器的抽象模拟。

程序计数器有时候也被称为为程序钩子

在JVM规范中,每个线程都有它自己的程序计数器,是线程私有的,生命周期与线程的生命周期保持一致。

2. 虚拟机栈

虚拟机栈

2.1 定义

Java Virtual Machine Stacks (Java 虚拟机栈)
栈

  • 每个线程运行时所需要的内存,称为虚拟机栈
    栈帧

  • 每个栈由多个栈帧(Frame)组成,对应着每次方法调用时所占用的内存
    在这里插入图片描述

  • 每个线程只能有一个活动栈帧,对应着当前正在执行的那个方法
    活动栈帧
    问题辨析

  1. 垃圾回收是否涉及栈内存?
    不需要,栈内存是方法调用时产生的栈帧内存,会在方法结束后弹出栈被自动回收
  1. 栈内存分配越大越好吗?
 	栈内存可以通过虚拟机参数指定 -Xss 

栈内存默认值

	栈内存越大,线程数越少
		假设500M的物理内存,每个栈内存1M那么可以有500个线程
		如果栈内存设置为2M那么只能有250个线程
	一般使用默认栈内存
  1. 方法内的局部变量是否线程安全?
    • 如果方法内局部变量没有逃离方法的作用访问,它是线程安全的
    • 如果是局部变量引用了对象,并逃离方法的作用范围 ,需要考虑线程安全

2.2 栈内存溢出

  1. 栈帧过多导致栈内存溢出(方法递归调用)
    栈帧过多导致栈内存溢出
  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回收就是内存泄露;

分类:

  • 经常发生: 发生内存泄露的代码会被多次执行,每次执行,泄露一块内存
  • 偶然发生: 在某些特定情况下才会发生
  • 一次性: 发生内存泄露的方法只会执行一次
  • 隐式泄漏: 一直占着内存不释放,直到执行结束; 严格的说这个不算内存泄漏,因为最终释放掉了, 但是如果执行时间特别长,也可能会导致内存耗尽

java内存泄露和内存溢出

4.3 堆内存诊断

  1. jps 工具
    查看当前系统中有哪些 java 进程
  2. jmap 工具
    查看堆内存占用情况 jmap - heap 进程id
  3. jconsole 工具
    图形界面的,多功能的监测工具,可以连续监测

案例

  • 垃圾回收后,内存占用仍然很高
    jdk8使用JVisualVM
    JVisualVM
    jdk9使用jmc
    jmc
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值