当前位置: 首页 > news >正文

深入理解 Java 内存区域与内存溢出异常

个人主页
文章专栏

文章目录

    • 一、引言
    • 二、Java 运行时数据区域
      • (一)程序计数器
      • (二)Java 虚拟机栈
      • (三)本地方法栈
      • (四)Java 堆
      • (五)方法区
      • (六)运行时常量池
      • (七)直接内存
    • 三、内存溢出异常实战
      • (一)Java 堆溢出
      • (二)虚拟机栈溢出
      • (三)方法区和运行时常量池溢出
      • (四)本机直接内存溢出
    • 四、结语

注:本文结合《深入理解JAVA虚拟机》第二章第一节和第二节而作。
在这里插入图片描述

一、引言

在 Java 编程领域,内存管理看似由虚拟机自动操持,开发者无需过度介入。然而,当内存泄漏或溢出问题悄然浮现,若对虚拟机内存运作机制缺乏深入认知,排查与修复工作将举步维艰。本文将深入剖析 Java 内存区域,并对常见的内存溢出异常进行详细探讨。

二、Java 运行时数据区域

(一)程序计数器

  • 功能:程序计数器是一块较小的内存空间,其作用是指示当前线程执行字节码的行号,是程序控制流的关键指示器,负责分支、循环、跳转等流程控制。
  • 特性:为线程私有,当线程执行 Java 方法时,记录字节码指令地址;执行本地方法时,计数器值为空。该区域不会出现 OutOfMemoryError

(二)Java 虚拟机栈

  • 线程私有性:Java 虚拟机栈同样是线程私有的,其生命周期与线程紧密相连。
  • 栈帧结构:以栈帧为单位存储方法执行时的局部变量表、操作数栈、动态连接及方法出口等信息。局部变量表存放基本数据类型和对象引用,编译期确定其大小。
  • 异常情况:可能抛出 StackOverflowError(当线程请求的栈深度超过虚拟机允许的深度)和 OutOfMemoryError(栈动态扩展时无法申请到足够内存)。

(三)本地方法栈

  • 功能与虚拟机栈的关系:功能类似虚拟机栈,主要为本地方法执行提供支持。
  • 实现方式:实现方式由虚拟机自行决定。
  • 异常抛出:在栈深度溢出或扩展失败时,会抛出 StackOverflowErrorOutOfMemoryError

(四)Java 堆

  • 地位与作用:是虚拟机管理的最大内存区域,被所有线程共享,用于存放对象实例,是垃圾收集的主要区域。
  • 分代收集理论:基于分代收集理论进行区域划分,可设置为固定或扩展大小。
  • 内存溢出情况:当内存不足且无法扩展时,抛出 OutOfMemoryError

(五)方法区

  • 线程共享性:线程共享区域,用于存储类型信息、常量、静态变量等。
  • 历史变迁:JDK 8 前常被称为 “永久代”,之后采用元空间实现。
  • 异常情况:内存不足时抛出 OutOfMemoryError

(六)运行时常量池

  • 所属区域:属于方法区,存放编译期生成的字面量与符号引用。
  • 动态性:具备动态性,运行时可添加常量。
  • 内存异常:内存申请失败时抛出 OutOfMemoryError

(七)直接内存

  • 特殊性质:并非虚拟机规范定义区域,NIO 可利用其分配堆外内存以提升性能。
  • 内存限制:不受 Java 堆大小限制,但受本机内存制约。
  • 溢出异常:超出限制时抛出 OutOfMemoryError

三、内存溢出异常实战

(一)Java 堆溢出

示例代码:通过不断创建对象耗尽堆内存。

import java.util.ArrayList;
import java.util.List;
class HeapOOM {static class OOMObject {}public static void main(String[] args) {List<OOMObject> list = new ArrayList<>();while (true) {list.add(new OOMObject());}}
}

解决思路:可通过 -Xmx 参数设置堆大小,溢出时抛出 OutOfMemoryError: Java heap space,可调整堆大小或优化对象创建逻辑来解决。

(二)虚拟机栈溢出

示例代码:利用无限递归使栈深度超限。

class StackSOF {private static void stackLeak() {stackLeak();}public static void main(String[] args) {try {stackLeak();} catch (Throwable e) {System.out.println("Stack depth: " + e.getMessage());e.printStackTrace();}}
}

解决方法:抛出 StackOverflowError,需检查递归算法并设置终止条件。

(三)方法区和运行时常量池溢出

示例代码:持续向常量池添加字符串。

import java.util.ArrayList;
import java.util.List;
class MethodAreaOOM {public static void main(String[] args) {List<String> list = new ArrayList<>();int i = 0;while (true) {list.add(String.valueOf(i++).intern());}}
}

应对策略:抛出 OutOfMemoryError,需关注常量池使用情况,避免无节制创建常量。

(四)本机直接内存溢出

示例代码:借助Unsafe类不断分配直接内存。

import sun.misc.Unsafe;
import java.lang.reflect.Field;
class DirectMemoryOOM {private static final int _1MB = 1024 * 1024;public static void main(String[] args) throws IllegalAccessException {Field unsafeField = Unsafe.class.getDeclaredFields()[0];unsafeField.setAccessible(true);Unsafe unsafe = (Unsafe) unsafeField.get(null);while (true) {unsafe.allocateMemory(_1MB);}}
}

解决方案:抛出 OutOfMemoryError,需合理配置虚拟机参数并监控直接内存使用。

四、结语

{
unsafe.allocateMemory(_1MB);
}
}
}


**解决方案**:抛出 `OutOfMemoryError`,需合理配置虚拟机参数并监控直接内存使用。## 四、结语深入掌握 Java 内存区域划分及内存溢出异常原理,是 `Java` 开发者进阶路上的关键。在日常开发中,应养成良好的内存管理习惯,借助工具监控内存使用,确保程序稳定、高效运行。

相关文章:

  • 研发效率破局之道阅读总结(2)流程优化
  • 【论文阅读】Orion: Online Backdoor Sample Detection via Evolution Deviance
  • 鸿蒙应用开发—鸿蒙app一键安装脚本
  • Vue3监听数据变化方法详解
  • 详解LeetCode中用字符串实现整数相加,字符串转整数及其溢出处理详解
  • 网络编程(UDP)
  • Flutter 应用在真机上调试的流程
  • HOW - 前端 sdk 实践(一)
  • 如何写好合同管理系统需求分析
  • 软考教材重点内容 信息安全工程师 第22章 网站安全需求分析与安全保护工程
  • 【C++算法】60.哈希表_字母异位词分组
  • PG中通过GIST创建其他自定义索引
  • 深度学习入门:神经网络的学习
  • 入门51单片机(1)-----点灯大师梦开始的地方
  • 网络安全与信息安全的区别​及共通
  • CEPH OSD_SLOW_PING_TIME_FRONT/BACK 警告处理
  • MuJoCo(Multi-Joint Dynamics with Contact)机器人仿真器存在的问题
  • rook-ceph 慢盘导致的 OSD_SLOW_PING_TIME_[BACK/FRONT] 告警分析
  • Elasticvue-轻量级Elasticsearch可视化管理工具
  • 25软考新版系统分析师怎么备考?重点考哪些?(附新版备考资源)
  • 招行一季度净利372.86亿降2.08%,营收降逾3%
  • 对话|贝聿铭设计的不只是建筑,更是生活空间
  • 苏州一季度GDP为6095.68亿元,同比增长6%
  • 挤占学生伙食费、公务考察到景区旅游……青岛通报5起违规典型问题
  • 日本大米价格连续16周上涨,再创最高纪录
  • 上海乐高乐园建设进入最后冲刺,开园限量纪念年卡将于5月开售