学习笔记——JVM

引言

什么是JVM

定义

Java Virtual Machine - Java程序得到运行时环境 (Java二进制字节码的运行环境)

JVM的主要特点:

  • 自适应编译器(Adaptive compiler): 一次编译,到处运行
  • 快速内存分配和垃圾收集(Rapid memory allocation and garbage collection)
  • 线程同步(Thread synchronization)

比较JDK、JRE和JVM

**1-1**  JDK、JRE、JVM比较

常见的JVM有哪些

**1-2**  常见JVM

学习路线

**1-3**  学习路线

内存区域

程序计数器

定义

Program Counter Register 程序计数器(寄存器),是当前线程所执行的字节码的行号指示器。
特点:

  1. 线程私有性。在多线程的情况下,程序计数器用于记录当前线程执行的位置,从而当线程被切换回来的时候能够知道该线程上次运行到哪儿了;
  2. 在Java虚拟机规范中,唯一一个不会存在内存溢出的区域

作用

寄存下一条JVM指令的地址,从而实现代码的流程控制。
**2-1**  程序计数器原理图

JAVA虚拟机栈

定义

Java Virtual Machine Stacks (Java虚拟机栈)

  • 每个线程(Thread)运行时需要的内存,称为虚拟机栈;
  • 每个栈由多个栈帧(Stack Frame)组成,对应每次方法调用是所占的内存;
  • 每个线程只能由一个活动栈帧 (即栈顶的栈帧),对应着正在执行的那个方法;
    **2-2**  Java虚拟机栈
问题辨析
  • 垃圾回收是否涉及栈内存?
    答:不涉及。因为栈帧随着方法调用而创建,随着方法结束而销毁。无论方法正常完成还是异常完成都算作方法结束。垃圾回收主要涉及堆空间内存。
  • 栈内存分配越大越好吗?
    答:不是。因为物理内存是固定的,栈内存划分越大,反而会让线程越少。
  • 方法内局部变量是否线程安全?
    • 如果局部变量没有逃逸方法内的作用范围,那么它是线程安全的;
    • 如果局部变量引用的对象,并逃逸方法的作用范围,那么需要考虑线程安全问题。

栈内存溢出

两种错误:

  • StackOverFlowError::如果线程请求的栈深度大于虚拟机所允许的深度,将会抛出该异常;
  • OutOfMemoryError:如果Java虚拟机栈容量可以动态扩展,当栈扩展时无法申请到足够的内存会抛出该异常

例如:

  • 栈帧过多导致栈内存溢出;(StackOverFlowError)
  • 关系映射时互相引用导致递归栈内存溢出;(StackOverFlowError)

线程运行诊断

案例1:CPU占用过多
定位:

  • top命令定位哪个进程对CPU占用过高;
  • ps H -eo pid,tid,%cpu | grep [进程id] (用ps命令定位哪个线程对cpu占用过高);
  • jstack [进程id],jstack命令输出该进程的所有线程运行状态,然后根据线程id找到有问题的线程,进一步找到问题代码的源码行号;

案例2:程序运行很长时间没有结果
使用jstack [进程id]命令追踪中断的进程中所有线程运行状态,在输出列表的尾部,找到有问题的线程,进一步定位到问题代码的源码行号。

本地方法栈

定义

Native Method Stacks (本地方法栈),和虚拟机栈所发挥的作用非常相似,区别是: 虚拟机栈为虚拟机执行 Java 方法 (也就是字节码)服务,而本地方法栈则为虚拟机使用到的 Native 方法服务。 在 HotSpot 虚拟机中和 Java 虚拟机栈合二为一。

堆(heap)

定义

Java 虚拟机所管理的内存中最大的一块,Java 堆是所有线程共享的一块内存区域,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例以及数组都在这里分配内存。

堆结构

JDK 7及以往版本的堆结构(来自官方文档)
**2-3** JDK 7及之前版本的堆结构

  • 年轻代是由eden和两个survivor空间组成。大多数对象最初是在eden中分配的。一个幸存者空间在任何时候都是空的,它用作eden中的任何活动对象的目的地,并在下一次复制收集期间用作其他幸存者空间。以这种方式在幸存者空间之间复制对象,直到它们老到可以被永久保存(复制到永久代)。
  • 永久代与Tenured代密切相关,作用是保存虚拟机所需的数据,以描述在Java语言级别上没有等效性的对象。例如,描述类和方法的对象存储在永久生成中。

JDK 8之后版本的堆结构:
2-4 JDK8之后的堆结构
与之前版本不同的是Permanent Generation(永久代)被Metaspace (元空间) 取代了。

堆内存溢出

常见错误:

java.lang.OutOfMemoryError: GC Overhead Limit Exceeded

当 JVM 花太多时间执行垃圾回收并且只能回收很少的堆空间时,就会发生此错误。

java.lang.OutOfMemoryError: Java heap space

假如在创建新的对象时, 堆内存中的空间不足以存放新创建的对象, 就会引发此错误。一般情况下,我们可以使用 -Xmx 设置最大堆空间参数,设置比默认值来的小的max size可以更早的暴露出错误。

堆内存诊断

  • jps工具:查看当前系统中有哪些Java程序;
  • jmap工具:查看堆内存占用情况,指令 jmap -heap [进程id];

注:由于JDK11中移除了jmap命令的-heap选项,需要使用jhsdb连接到Java进程:
jhsdb jmap [–pid pid | --exe executable --core coredump] [options]
jmap的options参照如上官方文档链接2-5 Options for the jmap Mode

  • jconsole工具:图形界面的,多功能的监测工具,可以连续监测;
  • jvisualvm( 虚拟机可视化工具 ), jdk9之后已被移除

方法区

定义

Method Area (方法区),它被所有线程所共享在JVM启动时被创建
方法区中主要存储了每一个class structure(类结构),例如run time constant pool(运行时常量池)、field(成员变量)、method data(方法数据)、code for method and constructors(方法和构造器代码)……

组成

2-6 方法区组成
图片来自: 黑马程序员JVM完整教程[22_方法区_定义]

JDK8以前,方法区隶属于堆,位于PermGen(永久代)中;JDK8及之后,永久代被移除,诞生了Meta Space(元空间,位于本地内存中),方法区隶属于Meta Space。
其中包括运行时常量池、类、类加载器等部分,从图中可以看出JVM 1.6时,String Table(字符串表,或称字符串常量池)位于永久代的常量池中;JVM 1.8时,String Table从方法区中分离出来,置于堆中。

方法区内存溢出

  • JDK 1.8以前会导致永久代内存溢出
    java.lang.OutOfMemoryError: PermGen space
    -XX:MaxPermSize=8m
  • JDK1.8以后会导致元空间内存溢出
    java.lang.OutOfMemoryError: Metaspace
    -XX:MaxMetaspaceSize=8m

场景:
spring
mybatis

运行时常量池

字符串常量池

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值