《深入理解Java虚拟机》读书笔记:第2章 Java内存区域与内存溢出异常

1、概述

Java虚拟机在执行Java程序的过程中会把它管理的内存区域划分成不同的区域。这些区域都有各自的用途以及创建和销毁时间,有的区域随着虚拟机进程的启动而存在,有的区域以来用户线程的启动而建立和销毁。Java虚拟机内存管理划分如下图:

在这里插入图片描述

2、程序计数器

程序计数器是一个较小的内存区域,它是线程私有的内存,可以看作是当前线程执行的字节码的行号指示器。一个处理器核心在同一时刻只能运行一个线程,程序计数器的设计是为了在切换线程的时候能够正确恢复线程的执行位置。此内存区域是唯一一个在Java虚拟机规范里没有规定任何OOM情况的区域。

如果当前执行的是Java方法,计数器记录的是正在执行的字节码指令的地址;如果正在执行的是Native方法,计数器的值是Undefined。

3、Java虚拟机栈

Java虚拟机栈也是线程私有的内存区域,它的生命周期与线程相同。Java虚拟机栈描述的是Java方法执行的内存模型,每个方法在执行的时候都会创建一个栈帧用于存放局部变量表、操作数栈、动态链接、方法出口等信息。每一个方法从调用到执行完成的过程对应一个栈帧入栈到出栈的过程。JVM规范规定了此内存区域的两种异常情况:

  1. 线程请求深度超过最大深度,跑出StackOverflowError
  2. 如果扩展时无法分配到足够大的内存,抛出OutOfMoryerror

4、本地方法栈

本地方法栈与Java虚拟机栈类似,不同的是Java虚拟机栈为Java方法服务,本地方法栈为Native方法服务。某些虚拟机(譬如Sum Hotspot虚拟机)直接把Java虚拟机栈和本地方法栈合二为一。可能的异常情况也与Java虚拟机栈一样。

5、 Java堆

Java堆(Heap)是虚拟机管理的内存中最大的一块,它被所有线程共享。此内存区域的唯一目的就是存放对象的实例。几乎所有对象实例都在这里分配内存,除了某些栈上分配和标量替换优化技术导致的一些特殊情况。
Java堆是GC管理的主要区域,所以有时候也叫GC堆。在内存不足以完成实例分配切无法扩展时,抛出OutOfMemoryError。

6、方法区

方法区也是线程共享的区域,用于存放已经被虚拟机加载的类的信息、常量、静态变量、即时编译器编译后的代码等数据。内存不足的时候也会抛出OOM异常。

方法去还有一个值得一提的部分是运行时常量池。Class文件除了有类的版本、字段、方法和接口等信息外,还有一项信息是常量池,用于存放编译期生成的各种字面量和符号引用。

7、直接内存

直接内存并不是虚拟机运行时数据区域的一部分,也不是JVM规范规定的内存区域,但这部分内存也被频繁使用,也可能导致OOM,所以一并讨论。
JDK1.4中新加入了NIO,引入了一种基于Channel和Buffer的I/O方式,它可以使用Native函数直接分配堆外内存,然后通过一个存储在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作。在某些场景下可以显著提高性能,避免在Java堆和Native堆来回复制数据。
本机直接内存不受Java堆大小的限制,受本机总内存大小、殉职空间大小的限制,也可能会抛出OOM异常。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值