方法区和堆

本文深入解析Java内存模型,包括方法区、堆、新生代、老年代和永久代的概念与作用。探讨了不同区域如何存储类信息、常量、对象实例等,并介绍了JDK1.7与1.8中方法区实现的演变。

1、方法区(Method Area),又称永久代(Permanent Generation),又称非堆区(Non-Heap space)

方法区是被所有线程共享。
所有字段和方法字节码,以及一些特殊方法如构造函数,接口代码也在此定义。
简单说,所有定义的方法的信息都保存在该区域,此区属于共享区间。

静态变量、常量、类信息(构造方法/接口定义) 、运行时常量池存在方法区中 。
但是实例变量 存在堆内存中,和方法区无关。

以上,只是逻辑上的定义。
在HotSpot中,方法区仅仅只是逻辑上的独立,实际上还是包含在Java堆中,也是就说,方式区在物理上属于Java堆区中的一部分,而永久区(Permanent Generation)就是方法区的实现。

2、堆 (heap)

一个JVM实例只存在一个堆内存,堆内存的大小是可以调节的。
类加载器读取了类文件后,需要把类、方法、常变量放到堆内存中,保存所有引用类型的真实信息,以方便执行器执行。

堆在逻辑上分为三部分 新生区 + 养老区 + 永久区
新生代(Young Generation,常称为YoungGen),位于堆空间;
老年代(Old Generation,常称为OldGen、TenuringGen),位于堆空间;
永久代(Permanent Generation,常称为PermGen),位于非堆空间

2.1、新生区(New/Young Generation)

新生代(Young Generation),常称为YoungGen,位于堆空间
新生区 又分为 Eden区 和 Survior(幸存区)。
Eden : 新创建的对象
Survior 0、1:经过垃圾回收,但是垃圾回收次数小于15次的对象

2.2、养老代(Old Generation)

老年代(Old Generation),常称为OldGen,位于堆空间
Old : 垃圾回收次数超过15次,依然存活的对象

2.3、永久区(Permanent Generation)

永久代(Permanent Generation),常称为 PermGen,位于非堆空间

永久区是一个常驻内存区域,用于存放JDK自身所携带的 Class,Interface 的元数据,也就是说它存储的是运行环境必须的类信息,被装载进此区域的数据是不会被垃圾回收器回收掉的,关闭 JVM 才会释放此区域所占用的内存。

如果出现 java.lang.OutOfMemoryError: PermGen space ,说明是Java虚拟机对永久代Perm内存设置不够。
一般出现这种情况,都是程序启动需要加载大量的第三方jar包。
例如:在一个Tomcat下部署了太多的应用。或者大量动态反射生成的类不断被加载,最终导致Perm区被占满。

方法区的实现的演变
Jdk1.7之前:hotspot虚拟机对方法区的实现为永久代 。
Jdk1.8及之后:hotspot移除了永久代用元空间(Metaspace)。

运行时常量池存和字符串常量池的变化
JDK1.7之前
运行时常量池(包含字符串常量池 )存放在方法区,此时 hotspot 虚拟机对方法区的实现为永久代

JDK1.7
字符串常量池被从方法区拿到了中;
运行时常量池剩下的东西还在方法区, 也就是hotspot中的永久代。

JDK1.8
hotspot移除了永久代,用元空间(Metaspace) 取而代之。这时候,
字符串常量池还在,
运行时常量池还在方法区, 只不过方法区的实现从永久代变成元空间(Metaspace)。

在Java虚拟机(JVM)的内存模型中,方法区是两个重要的内存区域,它们在内存分配、管理以及用途上有着明确的区分与联系。 ### 方法区的作用与特点 方法区在JVM中主要用于存储类的元数据信息,包括类的结构信息(如类名、访问修饰符、字段、方法等)、常量池、静态变量以及编译器编译后的代码。根据《Java虚拟机规范》,方法区在逻辑上属于的一部分,但它通常被称为“非”,以区别于Java中用于存储对象实例的区域。这种设计的目的是为了强调方法区的其他部分在功能上的差异[^1]。 在早期的JVM实现中,方法区通常与共享同一块内存区域,但随着JVM的发展,特别是在JDK 1.8中,方法区的实现被元空间(Metaspace)所取代,元空间从内存中独立出来,使用本地内存(Native Memory)进行存储。这一变化旨在解决方法区大小受限于内存的问题,并减少因类元数据过多而导致的内存溢出风险。 ### 的作用与特点 是JVM中用于存储对象实例的主要内存区域。所有通过`new`关键字创建的对象都会在上分配内存空间。被划分为不同的代(Generation),主要包括新生代(Young Generation)老年代(Old Generation),以便于垃圾回收器更高效地管理内存[^3]。 ### 方法区的交互 尽管方法区在逻辑上被认为是的一部分,但实际上,它们在功能管理上是相对独立的。方法区主要负责存储类的元数据,而则专注于对象实例的存储。在类加载过程中,类的元数据会被加载到方法区,而类的实例对象则会被分配到中。例如,当一个类被加载时,其结构信息会被存储在方法区,而该类的对象实例则会在中创建,并通过方法区中的类信息来初始化[^2]。 此外,JVM还支持一种优化技术——栈上分配(Stack Allocation),即如果一个对象的引用不会超出当前方法的作用域,那么JVM可以选择将这个对象分配到栈上,而不是上。这种方法可以减少内存的压力,并避免垃圾回收带来的性能开销。例如,在以下代码中,`Point`对象的引用仅限于`createObject()`方法内部,因此可以被分配在栈上: ```java public class StackAllocationExample { public static void main(String[] args) { for (int i = 0; i < 1000000; i++) { createObject(); } } public static void createObject() { // 这里创建的对象未发生逃逸 Point point = new Point(1, 2); } } class Point { private int x; private int y; public Point(int x, int y) { this.x = x; this.y = y; } } ``` ### 方法区的调用关系 当程序开始运行时,如`main`方法被调用,JVM会创建一个新的线程,并在该线程的虚拟机栈中创建一个栈帧,用于存储`main`方法的局部变量操作数栈。如果在`main`方法中创建了一个对象,如`MyObject obj = new MyObject();`,JVM会在上为这个对象分配内存,并在当前栈帧的局部变量表中存储这个对象的引用。如果`main`方法调用了另一个方法,如`obj.someMethod()`,JVM会在虚拟机栈中为`someMethod`创建一个新的栈帧,并将其压入栈顶。当`someMethod`执行完毕,它的栈帧会从虚拟机栈中弹出,控制权返回到`main`方法的栈帧。当`main`方法执行结束,JVM会结束该线程,此时该线程的虚拟机栈会被销毁。垃圾回收器会定期检查中的对象,回收那些不再被任何栈帧引用的对象,以释放内存空间[^3]。 ### 总结 综上所述,方法区在JVM中各自承担着不同的职责。方法区主要用于存储类的元数据,而则用于存储对象实例。虽然方法区在逻辑上属于的一部分,但它们在实现上通常是独立的。随着JVM的发展,方法区的实现也在不断演进,最终在JDK 1.8中被元空间所取代,进一步优化了内存管理[^1]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符  | 博主筛选后可见
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值