JVM内存模型

1 篇文章 0 订阅

JVM架构图

在这里插入图片描述

JVM内存模型 – JDK1.8

在这里插入图片描述

程序计数器

1:当前所执行的字节码行号指示器(逻辑)
2:改变计数器的值来获取下一条需要执行的字节码指令
3:和线程是一对一的关系即线程私有
4:对java方法技术,如果natice方法则技术器值为Undefined
5:不会发生内存泄漏

java虚拟机栈

1:java方法执行的内存模型
2:包含多个栈帧

本地方法栈

1:与虚拟机栈相似,主要作用于标注了native的方法

Java堆可以细分为:新生代和老年代;在细致一点的有Eden空间、From Survivor空间、To Survivor空间等。

这样划分的目的是为了使 JVM 能够更好的管理堆内存中的对象,可以根据跟个年代的特点采用最适当的收集算法。在新生代中,每次垃圾收集时都发现有大批的对象死去,只有少量存活,那就选用复制算法,只需要付出少量存活对象的复制成本就可以完成收集。而老年代中因为对象存活率高、没有额外空间对它进行分配担保,就必须使用"标记—整理"算法来进行回收。

下图是Java堆内存默认划分示意图:
在这里插入图片描述
————————————————
原文链接:https://blog.csdn.net/weixin_33568172/article/details/114117072

JVM三大性能调优参数 -Xms -Xmx -Xss的含义:(java -Xms128m -Xmx128M -Xss256K -jar xxx.jar)

  • Xss规定了每个线程虚拟机栈(堆栈)的大小
  • Xms堆的初始值
  • Xmx堆能达到的最大值

Java内存模型中堆和栈的区别 — 内存分配策略

  • 静态存储:编译时确定每个数据目标在运行时的存储空间需求
  • 栈式存储:数据区需求在编译时未知,运行时模块入口前确定
  • 堆式存储:编译时或运行时模块入口都无法确定,动态分配

java内存模型中堆和栈的区别

  • 联系:引用对象、数组时,栈里定义变量保存堆中目标的首地址
  • 管理方式:栈自动释放,堆需要GC
  • 空间大小:栈比堆小
  • 碎片相关:栈产生的碎片远小于堆
  • 分配方式:栈支持静态和动态分配,而堆仅支持动态分配
  • 效率:栈的效率比堆高

方法区

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

在Java8中移除了永生代,取而代之是元空间(Metaspace) 移除了永久代(PermGen),替换为元空间(Metaspace)

永久代中的 class metadata 转移到了 native memory(本地内存,而不是虚拟机);

永久代中的 interned Strings(字符串常量池) 和 class static variables 转移到了 Java heap;

永久代参数 (PermSize MaxPermSize) -> 元空间参数(MetaspaceSize MaxMetaspaceSize)

绝大部分 Java 程序员应该都见过 “java.lang.OutOfMemoryError: PermGen space” 这个异常。这里的 "PermGen space"其实指的就是方法区。不过方法区和“PermGen space”又有着本质的区别。前者是 JVM 的规范,而后者则是 JVM 规范的一种实现,并且只有 HotSpot 才有 “PermGen space”,而对于其他类型的虚拟机,如 JRockit(Oracle)、J9(IBM) 并没有“PermGen space”。由于方法区主要存储类的相关信息,所以对于动态生成类的情况比较容易出现永久代的内存溢出。最典型的场景就是,在 jsp 页面比较多的情况,容易出现永久代内存溢出。
————————————————
原文链接:https://blog.csdn.net/weixin_33568172/article/details/114117072

拓展信息 — java堆的结构,以及堆中的永久代

java堆不是数据结构意义上的堆(一种有序的树),而是jvm的堆,也即是运行时的数据区。所有类的实例和数组都是在堆上分配内存,它在JVM启动时被创建,对象所占的内存是由自动内存管理系统也就是垃圾回收器回收。

堆内存是由存活的对象以及死亡的对象组成的。存活的对象不会被垃圾回收器回收;死亡的对象是还没有被垃圾回收器回收的对象,等下一个周期回收

永久代:永久代主要存在类定义,字节码,和常量等很少会变更的信息。并且永久代不会发生垃圾回收,如果永久代满了或者超过了临界值,会触发完全垃圾回收(Full Gc)

而在java8中,已经移除了永久代,新加了一个叫做元数据区的native内存区
————————————————
原文链接:https://blog.csdn.net/qq_18433441/article/details/78231867

MetaSpace相比PermGen的优势

  • 字符串常量池存在永久代中,容易出现性能问题和内存溢出
  • 类和方法的信息大小难以确定,给永久代的大小指定带来困难
  • 永久代会为GC带来不必要的复杂性
  • 方便HotSport与其他JVM如Jrockit的集成

虚拟机栈详解

如图

在这里插入图片描述

局部变量表

包含方法执行过程中的所有变量

操作数栈

包含入栈、出栈、复制、交换、产生消费变量

jvm指令执行示意图

在这里插入图片描述

最后

  • 关于String:字符串常量池的介绍
    文档链接地址:https://segmentfault.com/a/1190000009888357
  • JDK 1.8 下的 java.lang.Class 对象和 static 成员变量在堆还是方法区?
    文档链接地址:
    https://blog.csdn.net/Xu_JL1997/article/details/89433916?utm_medium=distribute.pc_relevant.none-task-blog-baidujs_title-0&spm=1001.2101.3001.4242
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值