先放转载链接:https://www.cnblogs.com/nhdlb/p/12156962.html
Java:Java的堆区、栈区和方法区详解
Java内存空间理解
堆:****堆主要存放Java在运行过程中new出来的对象,凡是通过new生成的对象都存放在堆中,对于堆中的对象生命周期的管理由Java虚拟机的垃圾回收机制GC进行回收和统一管理。类的非静态成员变量也放在堆区,其中基本数据类型是直接保存值,而复杂类型是保存指向对象的引用,非静态成员变量在类的实例化时开辟空间并且初始化。所以你要知道类的几个时机,加载-连接-初始化-实例化。
栈:****栈主要存放在运行期间用到的一些局部变量(基本数据类型的变量)或者是指向其他对象的一些引用,因为方法执行时,被分配的内存就在栈中,所以当然存储的局部变量就在栈中咯。当一段代码或者一个方法调用完毕后,栈中为这段代码所提供的基本数据类型或者对象的引用立即被释放;
常量池:****常量池是方法区的一部分内存。常量池在编译期间就将一部分数据存放于该区域,包含基本数据类型如int、long等以final声明的常量值,和String字符串、特别注意的是对于方法运行期位于栈中的局部变量String常量的值可以通过 String.intern()方法将该值置入到常量池中。
静态域:****位于方法区的一块内存。存放类中以static声明的静态成员变量。
方法区:****是各个线程共享的内存区域,它用于存储class二进制文件,包含了虚拟机加载的类信息、常量、静态变量、即时编译后的代码等数据。它有个名字叫做Non-Heap(非堆),目的是与Java堆区分开。
需要特别注意的是:
方法区是线程安全的。由于所有的线程都共享方法区,所以,方法区里的数据访问必须被设计成线程安全的。例如,假如同时有两个线程都企图访问方法区中的同一个类,而这个类还没有被装入JVM,那么只允许一个线程去装载它,而其它线程必须等待 !
最后总结起来就是:
栈:****为即时调用的方法开辟空间,存储局部变量值(基本数据类型),局部变量引用。注意:局部变量必须手动初始化。
堆:****存放引用类型的对象,即new出来的对象、数组值、类的非静态成员变量值(基本数据类型)、非静态成员变量引用。其中非静态成员变量在实例化时开辟空间初始化值。更具体点,个人感觉非静态成员变量是放在堆的对象中。
方法区:****存放class二进制文件。包含类信息、静态变量,常量池(String字符串和final修饰的常量值等),类的版本号等基本信息。因为是共享的区域,所以如果静态成员变量的值或者常量值(String类型的值能够非修改,具体请查看博客)被修改了直接就会反应到其它类的对象中。
image.png
线程私有【栈】:虚拟机栈、本地栈(就是执行native方法 C的栈???)、程序计数器。
线程共享【堆、方法区】:堆存放生成的对象(栈帧中局部变量表只是持有对象的地址引用),方法区存放静态变量以及final 常量。
每个线程有很多栈帧,一个栈帧对应一个方法。
image.png
每个线程有自己的程序计数器,标明在cpu碎片时间中上次运行到了哪里。 线程中 局部变量表 存储生成的局部变量。操作数栈执行计算操作???。动态连接指向生成的具体多态对象。方法出口 抛出方法执行结果。
讲的很好的一篇文章:https://blog.csdn.net/lioncatch/article/details/106039968
image.png
1.栈中程序计数器标明程序运行位置,在下次抢占到CPU时间片运行后,继续执行线程。每个线程只有一个程序计数器。
2022年4月1日16:13:53
补充另外文章链接:https://mp.weixin.qq.com/s/YguU_XNpSTmvAkH_kKA7Tw
堆中划分为 Eden。s0,s1,老年代
划分和对象创建与GC有关
新生成的对象在Eden区
触发 Minor GC后,还 "幸存" 的对象移动到S0
再次触发Minor GC后,S0和Eden 中存活的对象被移动到S1中,S0清空
每次移动时,自动递增计数器,超过默认值时 (印象中是16),移动到老年代,如果Eden中没有足够内存分配,也将直接在老年代中分配内存
老年代中依靠Major GC









网友评论