Java中的内存划分

Java程序在运行时,需要在内存中分配空间。为了提高效率,就对数据进行了不同空间的划分,因为每一片区域都有特定的处理数据方式和内存管理方式。
      程序计数器:保证线程切换后能恢复到原来的执行位置
      虚拟机栈:(栈内存)为虚拟机执行java方法服务:方法被调用时创建栈帧->局部变量表->局部变量、对象引用
      本地方法栈:为虚拟机执行使用到的native方法服务
      堆内存:存放所有new出来的东西
      方法区:存储被虚拟机加载的类信息、常量、静态常量、静态方法等。

GC对它们的回收:
    内存取余中的程序计数器、虚拟机栈、本地方法栈这3个取余随着线程而生,线程而灭;栈中的栈帧随着方法的进入和退出而有条不紊地执行者出栈和入栈的操作,每个栈帧中分配多少内存基本是在类结构确定下来时就一直的。在这几个区域不需要过多考虑回收的问题,因为方法结束或者线程结束时,内存自然跟着回收了。

GC回收的主要对象:
   1.程序计数器(线程私有)
     每个线程拥有一个程序计数器,在线程创建时创建,指向下一条指令的地址,执行本地方法时,其值为undefined
为了线程切换后能够恢复到正确的执行位置,每条线程都有一个独立的程序计数器。
   2.java虚拟机栈(线程私有)
     每个方法被调用时都会创建一个栈帧,用于存储局部变量表、操作站、动态链接、方法出口等信息。局部变量表存放的是:编译器克制的基本数据类型、对象引用类型。
    每个方法被调用到执行完成的过程,就对应着一个栈帧在虚拟机中从入栈到出栈的过程。
     在Java虚拟机规范中,对这个区域规定了两种异常情况:
    (1)如果线程请求的栈深度太深,超出了虚拟机所允许的深度,就会出现stackOverFlowError(比如无限递归。因为每一层栈帧都占用一定空间,而Xss规定了栈的最大空间,超出这个值就会报错)
    (2)虚拟机栈可以动态扩展,如果扩展到无法申请足够的内存空间,会出现OOM
   3.本地方法栈
     (1)本地方法栈与java虚拟机栈作用非常相似,奇缺别是:java虚拟机栈是为虚拟机执行java方法服务的,而本地方法栈则为虚拟机使用到的Native方法服务
   4.java堆:即堆内存(线程共享)
     (1)堆是java虚拟机所管理的内存区域中的最大的一块,java堆是被所有线程共享的内存区域,在java虚拟机启动时创建,堆内存的唯一目的就是存放对象实例几乎所有的对象实例都在堆内存分配。
     (2)堆是GC管理的主要区域 从垃圾回收的角度, 由于现在的垃圾收集器都是采用的分代收集算法,因此java堆还可以初步细分为新生代和老年代。 
    (3)Java虚拟机规定,堆可以处于物理上不连续的内存空间中,只要逻辑上连续的即可。在实现上既可以是固定的,也可以是可动态扩展的。如果在堆内存没有完成实例分配,并且堆大小也无法扩展,就会抛出OutOfMemoryError异常。

     5、方法区:(线程共享) 
    (1)用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。 
     (2)Sun HotSpot虚拟机把方法区叫做永久代(Permanent Generation),方法区中最终要的部分是运行时常量池。

    6、运行时常量池: 
(1)运行时常量池是方法区的一部分,自然受到方法区内存的限制,当常量池无法再申请到内存时就会抛出OutOfMemoryError异常。
### Java虚拟机内存划分详解 Java虚拟机(JVM)的内存主要划分为以下几个部分:堆、栈、方法区、程序计数器和本地方法栈。以下是对每个部分的功能和用途的详细说明。 #### 1. 堆 (Heap) 堆是JVM中最大的一块内存区域,它是线程共享的全局内存,主要用于存储对象实例和数组。所有的对象都在堆上分配内存[^3]。 - **特点**:堆是垃圾回收的主要区域,因此也被称为“GC堆”。堆被进一步划分为新生代和老年代。 - **作用**: - 新生代(Young Generation):用于存放新创建的对象。 - 老年代(Old Generation):存放经过多次垃圾回收后仍然存活的对象。 - **引用示例**: ```java String str = new String("Hello World"); ``` 上述代码中,`str` 对象会被分配在堆中。 #### 2. 虚拟机栈 (Java Virtual Machine Stacks) 虚拟机栈是线程私有的内存区域,生命周期与线程相同。每个线程运行时都会创建一个虚拟机栈,栈由多个栈帧组成,每个栈帧对应一次方法调用[^1]。 - **特点**:栈的特点是先进后出(LIFO),局部变量存储在此处[^2]。 - **作用**: - 存储方法执行过程中的局部变量、操作数栈、动态链接等信息。 - 每个方法执行时都会创建一个栈帧,方法执行完毕后栈帧弹出。 - **异常**:如果线程请求的栈深度大于虚拟机允许的深度,则会抛出 `StackOverflowError`;如果栈扩展超出限制,则会抛出 `OutOfMemoryError`。 #### 3. 方法区 (Method Area) 方法区是线程共享的内存区域,用于存储已被虚拟机加载的类信息、常量、静态变量以及编译后的代码等数据[^3]。 - **特点**:方法区通常实现为永久代(PermGen)或元空间(Metaspace),具体取决于JVM实现。 - **作用**: - 存储类的结构信息,如字段、方法数据、方法代码等。 - 存储运行时常量池(Runtime Constant Pool),这是方法区的一部分,用于存放编译期生成的各种字面量和符号引用。 - **异常**:如果方法区无法满足新的内存分配需求,则会抛出 `OutOfMemoryError`。 #### 4. 程序计数器 (Program Counter Register) 程序计数器是一块较小的内存区域,用于记录当前线程所执行的字节码指令的位置。 - **特点**:每个线程都有独立的程序计数器,它是线程私有的。 - **作用**:指示当前线程正在执行的指令地址。如果线程正在执行的是一个Java方法,则计数器记录的是JVM字节码指令的地址;如果是Native方法,则计数器值为空(Undefined)。 #### 5. 本地方法栈 (Native Method Stacks) 本地方法栈与虚拟机栈类似,但它服务于本地方法(Native Method)。本地方法通常是用其他语言(如C/C++)编写的函数。 - **特点**:与虚拟机栈类似,但专门为本地方法服务。 - **作用**:存储本地方法执行所需的内存。 --- ### 总结 Java虚拟机的内存模型主要包括堆、栈、方法区、程序计数器和本地方法栈五个部分。各部分的功能如下: - **堆**:存储对象实例和数组。 - **虚拟机栈**:存储方法执行过程中的局部变量和操作数。 - **方法区**:存储类的结构信息、常量和静态变量。 - **程序计数器**:记录当前线程的指令地址。 - **本地方法栈**:为本地方法提供内存支持。 ```java // 示例代码 public class MemoryExample { public static void main(String[] args) { int a = 10; // 局部变量a存储在虚拟机栈中 String str = new String("Hello"); // 字符串对象存储在堆中 } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值