JVM内存结构

本文详细介绍了JVM的内存结构,包括方法区、堆、栈、程序计数器和本地方法栈。方法区用于存储类信息,如运行时常量池、静态变量等。堆是对象的存储区域,新生代和老年代有不同的垃圾回收策略。栈为线程运行提供内存空间,局部变量的线程安全性取决于其是否为私有。程序计数器记录执行位置,而本地方法栈服务于本地方法。文章还讨论了JVM的相关参数配置及其意义。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

每个区域的作用:

  • 方法区(线程共享)

存放类的二进制字节码文件(二进制字节码文件中包括:类基本信息,常量池,类方法定义,包含了虚拟机指令);

所有java虚拟机共享的区,存储了和类的结构、类信息

类结构:运行时常量池(含字符串常量),静态变量,类的信息,常量

类信息:魔数,版本号,常量池,类(字段和方法),父类和接口数组,字段,方法等信息。

可以结合类加载器理解:所有上述的信息在Class对象中都可以获取到

方法区逻辑上是堆的一部分,不同厂商在具体实现时不一定,方法区的位置不固定,方法区会产生内存溢出

方法区是JVM中的一个规范,永久代和元空间是方法区的两个不同实现;

Jdk1.8之前使用永生代实现,1.8之后使用元空间实现。

  • 堆(线程共享)

通过new关键字创建的对象都会由堆存储;堆中不再被引用的对象会通过垃圾回收机制自动回收

细分一点的内存结构:年轻代、老年代;垃圾回收策略也不同。

逻辑上又分为:新生代(Eden、from、to)、老年代

  • jvm栈(线程私有)

线程运行时需要的内存空间;

每个线程只能有一个活动栈帧,对应着一个线程正在执行的那个方法,该栈帧处于栈顶。

一个栈内是由多个栈帧组成的,一个栈帧对应着一个方法的调用,每个栈帧都有自己独立的存储空间

问题辨析:

垃圾回收是否涉及栈内存?

 不涉及,栈帧在每次方法执行完毕会自动排出栈,即被回收。

栈内存分配越大越好吗?

可通过-Xss 改变栈内存分配大小;栈内存分配大了更多的是能够执行的方法更多了,那么相对的能够分配给线程的栈数量就减少了。

方法内的局部变量是否线程安全?

其实就是看变量是共享还是私有的

如果方法内局部变量没有逃离方法的作用访问,即私有的,则是线程安全;

如果局部变量引用了对象,并逃离了方法的作用,需要考虑线程安全

  • 程序计数器(线程私有)

为了线程切换后能恢复到正确的执行位置,每个线程有一个程序计数器,物理上程序计数器是通过寄存器(CPU组件中读取速度最快的单元)来实现的,程序计数器是线程私有的,是java内存中唯一不会存在内存溢出的区

  • 本地栈(线程私有)

本地方法栈的功能特点类似于虚拟机栈,每个线程单独创建自己的本地方法区,主要用于保存本地方法信息。

不同处:虚拟机栈服务的是java方法,本地方法栈服务的是Native方法。

本地方法举例:hashCode,notify,wait

常见参数

堆:‐Xms2048M ‐Xmx2048M

堆-新生代:‐Xmn1024M

栈:‐Xss512K

方法区:‐XX:MetaspaceSize=256M ‐XX:MaxMetaspaceSize=256M 

为什么最大与最小值设置成一样的?

参考链接:

JVM内存结构_好久啦的博客-优快云博客_jvm内存结构​​​​​​JVM学习之-JVM内存结构 - 大漠驼铃 - BlogJava

jvm调优技巧 - 内存抖动 、Xms和Xmx参数为什么要设置相同的值_java叶新东老师的博客-优快云博客_xms和xmx为什么要相同

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值