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