JVM内存模型

运行时数据区域划分

方法区、堆区、虚拟机栈、本地方法栈、程序计数器

JDK1.8前:线程共享(Heap堆区、方法区)、线程私有(虚拟机栈、本地方法栈、程序计数器)

JDK1.8后:线程共享(Heap堆区、元空间)、线程私有(虚拟机栈、本地方法栈、程序计数器)

程序计数器

是当前线程所执行的字节码的行号指示器

每当需要执行一条字节码指令时,就通过改变程序计数器的值来完成。程序中的分支、循环、跳转、异常处理、线程恢复等功能都需要依赖这个计数器来完成

不会出现内存溢出,随线程的创建而创建,随线程的死亡而死亡

Java 虚拟机栈(VM Stack)

由很多栈帧组成,每个栈帧由局部变量、操作数栈、动态链表和方法返回四部分组成虚拟机栈是线程私有区域,并且栈帧不允许被其他线程访问,不存在线程安全和垃圾回收问题。

可能会出现错误:栈溢出和内存溢出

虚拟栈通过-Xss设置参数

本地方法栈

可能会出现错误:栈溢出和内存溢出

用于存放对象实例和数字的内存区域

"几乎"所以的对象实例以及数组都在这里分配空间

堆区的组成:新生代+老年代

新生代(1/3)、老年代(2/3)目的为了更好的会内存,或更快分配内存

新生代分为伊甸区和幸存区(8:1:1)--幸存区放Minor GC和没有被回收的对象

老年代:放Minor GC之后且年龄计数器达到15依然存活的对象、Major GC和Full GC之后依然存活的对象。

堆空间的大小设置

-Xms:初始堆内存

-Xmx:最大堆内存

-Xmn:新生代内存

创建对象的内存分配

对象在伊甸区生成,当伊甸区填满的时候,会触发YGC,YGC垃圾回收的时候,在伊甸区实现清楚策略,没有被引用的对象直接回收。

依然存活的对象会被移送到幸存区(分为S0、S1),当YGC时候,将存活对象复制到未使用的幸存空间,将正在使用的空间完全清除,交换两块空间的使用状态,每次交换,年龄+1。

如果YGC要移送的对象大于幸存区容量的上限,则直接交给老年代。一个对象不会永远在新生代,在JVM中一个对象从新生代晋升到老年代的阈值是15,可以在幸存区交换14次后,晋升到老年代。

堆区的分代垃圾收集思想
部分收集

新生代:YGC、老年代:FGC、混合收集:G1收集器

整堆收集

FGC:回收整个Java堆区,默认堆空间达到80%会触发。很少会触发,基本是10天半个月。

GC垃圾回收的影响

耗时太长、次数太多会影响进程的性能,导致进程响应变慢,或无法响应。

产生FGC原因

大对象、内存泄漏、程序频繁生成一些长生命周期的对象(存活年龄超过分代年龄便会进入老年代)、程序BUG倒是动态生成了很多新类、JVM参数设置不合理。

产生错误

内存溢出(堆内存不足、JVM花时间太长,只能回收很少堆空间)

元空间

存放类信息、常量、静态变量、JIT即时编译器编译后的机器代码等数据。

1.6:方法区

1.7:将字符串常量池、静态变量转移到了堆区。

1.8:元空间

字符串常量池

第一种方式是在常量池中获取字符串对象;

第二种方式是直接在堆内存空间创建一个新的字符串对象;

String 的intern( )方法:

检查指定字符串在常量池中是否存在?如果存在,则返会地址,如果不存在,则在常量池中创建。

相关推荐
超龄超能程序猿15 小时前
Spring AI Alibaba 与 Ollama对话历史的持久化
java·人工智能·spring
hrrrrb15 小时前
【Spring Security】认证(二)
java·后端·spring
33255_40857_2805915 小时前
告别密码爆破!手把手教你用注解和拦截器实现登录限流
java
舒克日记15 小时前
基于springboot针对老年人的景区订票系统
java·spring boot·后端
【杨(_> <_)】15 小时前
SAR信号处理重要工具-傅里叶变换(二)
算法·信号处理·傅里叶分析·菲涅尔函数
怎么没有名字注册了啊15 小时前
爬动的蠕虫
算法
取酒鱼食--【余九】15 小时前
机器人学基础(一)【坐标系和位姿变换】
笔记·算法·机器人·开源·机器人运动学·机器人学基础
西西学代码15 小时前
Flutter---showCupertinoDialog
java·前端·flutter
多多*15 小时前
上传文件相关业务,采用策略模式+模版方法模式进行动态解耦
java·开发语言