内存区域与内存溢出异常

2024-08-28 14:44
文章标签 异常 内存 区域 溢出

本文主要是介绍内存区域与内存溢出异常,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

内存区域与内存溢出异常

内存区域

程序计数器

程序计数器就是当前线程的字节码的行号指示器,是程序控制流的指示器,分支,循环,跳转,异常处理,均要这个计数器来完成。

Java虚拟机的多线程是线程轮流切换,分配执行时间来实现的,所以在任何一个时刻,一个处理器只会执行一条指令,执行完后会依赖程序计数器恢复到正确的执行位置,每条线程都有一个独立的程序计数器,放置他们的内存是线程私有的内存,其互不影响,独立存储。

Java虚拟机栈

在每个方法被执行的时候,Java虚拟机都会创建一个栈帧用于存储局部变量表,操作数栈,动态连接,方法接口等。每一个方法开始执行到结束都对应着一个栈帧从入栈到出栈的过程。

Java虚拟机的内存区域被笼统的分为堆内存和栈内存两部分,栈内存通常就指局部变量表。

局部变量表存放了编译期各种可知的Java虚拟机基本数据类型或是对象引用和returnAddress类型。

这些变量存储空间以局部变量槽来表示,其中64位长度的long和double会占用两个槽,其余都是一个槽,局部变量表的内存空间在编译期间完成分配,当进入一个方法的时候,这个栈帧要分配多少局部变量内存空间是完全确定的。方法运行期间不会改变内存空间的大小。

Java虚拟机对该区域规定了两个异常,一个是线程请求的深度大于虚拟机所允许的深度引发StackOverflowError异常;若虚拟机容量可以动态拓展,当栈拓展到无法申请到足够的内存时会抛出OutOfMemoryError异常。

本地方法栈

本地方法栈与虚拟机栈类似,本地方法栈是为虚拟机使用到的本地方法服务。

《Java虚拟机规范》当中没有对本地方法栈有明确的规定,因此任何虚拟机会根据需要自由使用它,甚至有些虚拟机将本地方法栈和Java虚拟机栈合二为一,比如说HotSpot虚拟机。

异常情况和Java虚拟机栈相同。

Java堆

《Java虚拟机规范》当中表示”所有的对象实例以及数组都应当在堆上分配“。但是现在逃逸分析技术的强大,栈上分配,标量替换导致这句话也是不那么绝对了。

Java堆是垃圾收集器管理的区域,也就是GC堆,从分配内存的角度来看,所有线程共享的Java堆都可以分配出多个线程私有的分配缓冲区(Thread Local Allocation Buffer,TLAB)以提升对象分配的效率,将Java堆进行细化只是为了更好的回收内存,或是更好的分配内存。

《Java虚拟机规范》规定Java堆可以处于物理不连续的内存空间当中,但是在逻辑上他应该被视为连续的内存空间,多数虚拟机出于实现简单、存储高效的考虑会设计为连续的内存空间。

Java堆可以是固定内存的,也可以是可扩展的,当前主流的Java虚拟机都是按照可扩展来设计的,若堆无法扩展,对象实例无法分配会抛出OutOfMemory的异常。

方法区

方法区和Java堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类型信息,常量,静态变量,即使编译器编译后的代码缓存等数据。

运行时常量池

运行时常量池是方法区的一部分,在编译的时候将生成的各种字面量与符号引用,将这部分加载后存放到方法区的运行时常量池中。在方法执行的过程当中,如果出现新的常量,也会被放入运行时常量池当中。

直接内存

直接内的出现是因为在JDK1.4当中引入了NIO类,引入了一种基于通道与缓冲区的IO方式,他可以使用Native函数库直接分配堆外内存,然后通过一个储存在Java堆当中的DirectByteBuffer对象来作为这块内存的引用进行操作。

本机的直接内存分配不会收到Java堆的限制,但是在按照实际内存去设置虚拟机参数信息时,会经常忽略掉直接内存,使得各个内存区域总和大于物理内存限制。

对象的内存布局

在HotSpot虚拟机里,对象分为三个部分:对象头(Mark word+指向对象的指针)+实例数据(真实有效的信息)+对齐填充(总大小保持8的倍数)。

对象的访问定位

Java程序会通过栈上的reference数据来操作堆上的具体对象。有两种方式访问,一个是通过句柄访问,一个是按照直接指针访问

在这里插入图片描述

句柄访问会直接划分出一块内存来作为句柄池,reference在存储的就是对象的句柄地址,句柄当中存储了对象实例数据与类型数据各自具体地址信息。

直接指针访问,Java堆就得考虑如何放置访问类型数据的相关信息、reference当中存储的就是对象地址,直接访问即可。

异常处理

OutOfMemory异常

  1. Java堆溢出:Java堆用于储存对象实例,一直添加对象的时候就会触发这个异常,通过参数-XX:+HeapDumpOnOutOfMemoryError可以让虚拟机再出现内存溢出异常的时候Dump出当前内存堆转储快照来进行分析。有两种第一个是内存溢出,另一个是内存泄漏,若是内存泄漏就可以通过GC Roots引用链找到泄漏对象,进而找到具体位置。
  2. 虚拟机栈和本地方法栈异常:一个是线程请求的深度大于虚拟机所允许的深度引发StackOverflowError异常;若虚拟机容量可以动态拓展,当栈拓展到无法申请到足够的内存时会抛出OutOfMemoryError异常。
  3. 本机直接内存溢出:内存无法分配导致。有一个明显特征是Dump文件一很小,而程序又间接使用了DirectMemory。

这篇关于内存区域与内存溢出异常的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1115065

相关文章

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

关于Java内存访问重排序的研究

《关于Java内存访问重排序的研究》文章主要介绍了重排序现象及其在多线程编程中的影响,包括内存可见性问题和Java内存模型中对重排序的规则... 目录什么是重排序重排序图解重排序实验as-if-serial语义内存访问重排序与内存可见性内存访问重排序与Java内存模型重排序示意表内存屏障内存屏障示意表Int

如何测试计算机的内存是否存在问题? 判断电脑内存故障的多种方法

《如何测试计算机的内存是否存在问题?判断电脑内存故障的多种方法》内存是电脑中非常重要的组件之一,如果内存出现故障,可能会导致电脑出现各种问题,如蓝屏、死机、程序崩溃等,如何判断内存是否出现故障呢?下... 如果你的电脑是崩溃、冻结还是不稳定,那么它的内存可能有问题。要进行检查,你可以使用Windows 11

Python中异常类型ValueError使用方法与场景

《Python中异常类型ValueError使用方法与场景》:本文主要介绍Python中的ValueError异常类型,它在处理不合适的值时抛出,并提供如何有效使用ValueError的建议,文中... 目录前言什么是 ValueError?什么时候会用到 ValueError?场景 1: 转换数据类型场景

Spring中Bean有关NullPointerException异常的原因分析

《Spring中Bean有关NullPointerException异常的原因分析》在Spring中使用@Autowired注解注入的bean不能在静态上下文中访问,否则会导致NullPointerE... 目录Spring中Bean有关NullPointerException异常的原因问题描述解决方案总结

Python中的异步:async 和 await以及操作中的事件循环、回调和异常

《Python中的异步:async和await以及操作中的事件循环、回调和异常》在现代编程中,异步操作在处理I/O密集型任务时,可以显著提高程序的性能和响应速度,Python提供了asyn... 目录引言什么是异步操作?python 中的异步编程基础async 和 await 关键字asyncio 模块理论

详解Python中通用工具类与异常处理

《详解Python中通用工具类与异常处理》在Python开发中,编写可重用的工具类和通用的异常处理机制是提高代码质量和开发效率的关键,本文将介绍如何将特定的异常类改写为更通用的ValidationEx... 目录1. 通用异常类:ValidationException2. 通用工具类:Utils3. 示例文

无人叉车3d激光slam多房间建图定位异常处理方案-墙体画线地图切分方案

墙体画线地图切分方案 针对问题:墙体两侧特征混淆误匹配,导致建图和定位偏差,表现为过门跳变、外月台走歪等 ·解决思路:预期的根治方案IGICP需要较长时间完成上线,先使用切分地图的工程化方案,即墙体两侧切分为不同地图,在某一侧只使用该侧地图进行定位 方案思路 切分原理:切分地图基于关键帧位置,而非点云。 理论基础:光照是直线的,一帧点云必定只能照射到墙的一侧,无法同时照到两侧实践考虑:关

NameNode内存生产配置

Hadoop2.x 系列,配置 NameNode 内存 NameNode 内存默认 2000m ,如果服务器内存 4G , NameNode 内存可以配置 3g 。在 hadoop-env.sh 文件中配置如下。 HADOOP_NAMENODE_OPTS=-Xmx3072m Hadoop3.x 系列,配置 Nam

Thymeleaf:生成静态文件及异常处理java.lang.NoClassDefFoundError: ognl/PropertyAccessor

我们需要引入包: <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-thymeleaf</artifactId></dependency><dependency><groupId>org.springframework</groupId><artifactId>sp