Java内存区域分布

概述

JVM内存 = 堆内存 + 线程数量 * 栈内存
在这里插入图片描述在这里插入图片描述
1、方法区和堆为线程共享区域
2、虚拟机栈、本地方法栈、程序计数器为线程隔离区域,每一个线程都是独立的

java内存可分为:

程序计数器

程序计数器(Program Counter Resister)是一块较小的内存空间,可以看做是当前线程所执行的字节码行号指示器。虚拟机的字节码解释器工作时就是通过改变这个计数器的值来选取吓一跳执行的字节码的指令、分支、循环、跳转、异常处理、线程恢复等基础的功能都需要依赖这个计数器来完成。

由于Java虚拟机的多线程是通过线程轮流切换并分配处理器的执行时间来实现的,在任何的时刻,一个处理器都只会执行一条线程中的指令,因此每条线程都需要一个独立的程序计数器,各条线程之间的计数器互不影响,独立存储。

Java虚拟机栈

Java虚拟机栈(Java Virtual Machine Stacks)也是线程私有的,生命周期与线程相同。虚拟机栈描述的是Java方法执行的内存模型,每个方法在执行的同时都会创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等信息。

局部变量表存放了编译器可知的各种基本数据类型(boolean、byte、double、float、int、long、char、short)、对象引用和returnAddress类型(指向了一条字节码指令的地址)。

栈是有长度限制的,在这个区域有两种异常:如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常;如果虚拟机栈可以动态扩展,如果扩展时无法申请到足够的内存,就会抛出OutOfMemoryError异常。

本地方法栈

本地方法栈(Native Method stack)和虚拟机栈是相似的,区别在于虚拟机栈为虚拟机执行Java方法,而本地方法则为虚拟机使用到的Native方法。

Java堆

Java堆(Java Heap)是Java虚拟机内存中最大的一块,该区域存放对象实例,几乎所有的对象实例都在这里分配。Java堆也是垃圾收集器管理的主要区域,现在的垃圾收集器基于分带收集算法(新生代、老生代)。

Java堆可以处于物理上不连续的内存空间中,可以通过-Xmx和-Xms来控制大小,如果在堆中没有完成实例的分配,并且堆无法再扩展时,将会抛出OutOfMemoryError异常。

方法区

方法区(Method Area)与Java堆一样,是各个线程共享的区域,用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。

运行常量池

运行常量池(Runtime Constant Pool)是方法区的一部分,Class文件中除了有类的版本、字段、方法、接口等描述信息之外,还有一项信息是常量池,用于存放编译器生成的各种字面量和符号引用。

运行时的常量池相对于Class文件的常量池有一个重要的特性是具备动态性,Java语言并不要求常量一定只有编译期才能产生,在运行期间也可以将新的常量放入池中,这种特性被开发人员利用得比较多的是String类的intern()方法。

直接内存

直接内存(Direct Memory)并不是虚拟机运行数据区的一部分,也不是Java虚拟机规范中的内存区域,但是这部分内存也被频繁使用,而且也可能导致OutOfMemoryError异常。

在JDK1.4中新加入了NIO(New Input/Output)类,引入了一种基于通道与缓冲区的I/O方式,它可以使用Native函数库直接分配堆外内存,然后通过存储在Java堆中的DirectByteBuffe对象作为这块内存的引用进行操作,这样可以显著的提高性能,因为这样可以避免在Java堆和Native堆来回的复制数据。

在这里插入图片描述

### Java JVM 内存区域分布详解 #### 1. 堆内存(Heap Memory) 堆内存是JVM管理的主要内存部分之一,主要用于存储对象实例。当`java.lang.OutOfMemoryError: Java heap space`发生时,意味着在创建新的对象时,堆内存中的空间不足以存放新创建的对象[^1]。 - **年轻代(Young Generation)** - 年轻代分为Eden区和两个Survivor区(S0和S1)。大多数情况下,新对象会在Eden区中分配。一旦Eden区满,则会触发一次Minor GC来清理该区域内的无用对象并移动存活对象到其中一个Survivor区。随着新生代空间占用率升高,Minor GC频率也会增加[^2]。 - **老年代(Old Generation)** - 经过多次Minor GC仍然存活下来的对象会被晋升至老年代。这里发生的GC称为Major GC或Full GC,通常涉及整个堆空间的扫描与整理,耗时较长。 #### 2. 方法区(Method Area) 方法区用来保存已被虚拟机加载的类信息、常量池、静态变量以及即时编译器编译后的代码等数据。自Java 8起,永久代被移除而由元空间替代;后者位于原生内存而非堆内,并且其容量仅受限于操作系统所能提供的最大可用内存范围[^3]。 - **永久代(Permanent Generation,在Java 7及更早版本存在)** - **元空间(Metaspace,从Java 8开始引入)** #### 3. 运行时常量池(Runtime Constant Pool) 这是方法区内的一部分特殊区域,用于存放编译期间产生的各类字面量和符号引用。对于字符串而言,还特别设有一个名为“字符串常量池”的子集,旨在优化相同内容字符串之间的共享机制以节省资源开销[^4]。 ```java // 字符串常量池示例 String str1 = "hello"; String str2 = "hello"; // 实际上str1 和 str2指向同一个地址 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值