【Java】虚拟机(JVM)内存模型全解析

目录

一、运行时数据区域划分

版本的差异:

二、程序计数器

程序计数器主要作用

三、Java虚拟机

1. 虚拟机运行原理

2. 活动栈被弹出的方式

3. 虚拟机栈可能产生的错误

4. 虚拟机栈的大小

四、本地方法栈

五、堆

1. 堆区的组成:新生代+老生代

2. 堆空间的大小设置

3. 创建对象的内存分配

4. 堆区产生的错误

六、字符串常量

1. String 的两种创建方式

2. String 的intern()方法:

3. String的拼接

4. String s1 = new string("abc");这句代码创建了几个字符串对象?


        

        JVM 虚拟机在执行 Java 程序的过程中,会把它管理的内存划分成若干个不同的区域,每个区域有各自的不同的用途、创建方式及管理方式。有些区域随着虚拟机的启动一直存在,有些区域则随着用户线程的启动和结束而建立和销毁,这些共同组成了 Java 虚拟机的运行时数据区域,也被称为JVM内存模型

一、运行时数据区域划分

        JVM 虚拟机在执行 Java 程序的过程中会把它管理的内存划分成若干个不同的数据区域。由方法区堆区虚拟机栈本地方法栈程序计数器五部分组成。

        

版本的差异:

        JDK 1.8 之分为:线程共享( Heap 堆区、Method Area 方法区)、线程私有(虚拟机栈、本地方法栈、程序计数器)。

        JDK 1.8 以分为:线程共享( Heap 堆区、MetaSpace 元空间)、线程私有(虚拟机栈、本地方法栈、程序计数器)。       

                                                               ​​​​​​        

        其中虚拟机栈、本地方法栈、程序计数器是线程私有的区域,所以随着线程消亡而结束。而线程共享的 Heap 堆区、 Metaspace 元空间会随着虚拟机的启动,一直存在

二、程序计数器

        程序计数器是一块较小的内存空间,是当前线程所执行的字节码的行号指示器。

程序计数器主要作用

  • 字节码解释器通过改变程序计数器来依次读取指令,从而实现代码的流程控制,如:顺序执行、选择、循环、异常处理。
  • 在多线程的情况下,程序计数器用于记录当前线程执行的位置,从而当线程被切换回来的时候,能够知道当前线程的运行位置,恢复当前线程的执行。

        程序计数器是唯一一个不会出现 OutofMemoryError 的内存区域,它随着线程的创建而创建,随着线程的结束而死亡。

三、Java虚拟机

        虚拟机栈是线程执行 Java 程序时,处理 Java 方法中内容的内存区域。虚拟机栈也是线程私有的区域,每个 Java方法被调用的时候,都会在虚拟机栈中创建出一个栈帧,而每个栈帧又由局部变量表操作数栈动态链接方法返回四部分组成,有些虚拟机的栈帧还包括了一些附加信息。

        JVM 内存区域可以粗略的区分为堆内存(Heap)和栈内存(stack)。其中栈就是 JVM Stack 虚拟机栈,或者说是虚拟机栈中局部变量表部分,局部变量表主要存放了编译期可知的各种基本数据类型变量值( boolea、byte、char、short、int、float、long、double )、对象引用(reference 类型,它不同于对象本身,可能是一个指向对象起始地址的引用指针,也可能是指向一个代表对象的句柄或其他与此对象相关的位置)。

1. 虚拟机运行原理

        每一次方法调用都会有一个对应的栈帧被压入 JVM Stack 虚拟机栈,每一个方法调用结束后,代表该方法的栈帧会从 JVM Stack 虚拟机栈中弹出。虚拟机栈是线程的私有区域,并且栈帧不允许被其他线程访问,所以不存在线程安全问题,栈帧弹出后就内存就会被系统回收,所以不也存在垃圾回收问题。

        在活动线程中,只有位于栈顶的帧才是有效的,称为当前活动栈帧,代表正在执行的当前方法。在 JVM 执行引擎运行时,所有指令都只能针对当前活动栈帧进行操作。虚拟机栈通过 pop 和 push 的方式,对每个方法对应的活动栈帧进行运算处理,方法正常执行结束,肯定会跳转到另一个栈帧上。

2. 活动栈被弹出的方式

        Java 方法有两种返回方式,不管哪种返回方式都会导致当前活动栈帧被弹出:

  • return 语句
  • 抛出异常

3. 虚拟机栈可能产生的错误

        Java 虚拟机栈会出现两种错误:StackOverFlowErrorOutOfMemoryError

  • StackOverFlowError:当线程请求栈的深度超过 JVM 虚拟机栈的最大深度的时候,就抛出 StackOverFlowError 错误。
  • OutofMemoryError:JVM 的内存大小可以动态扩展,如果虚拟机在动态扩展栈时无法申请到足够的内存空间,则抛出 OutOfMemoryError 异常。

4. 虚拟机栈的大小

        虚拟栈的大小可以通过 -Xss 参数设置,默认单位是 byte,也可以使用k,m,g作为单位(不区分大小写)。例如:-xss 1m

        在不同操作系统下的 -xss 默认值不同:

  •         Linux:1024k
  •         MacOs:1024k
  •         Windows:默认值依赖于虚拟机的内存

四、本地方法栈

        native 关键字修饰的本地方法被执行的时候,在本地方法栈中也会创建一个栈帧,用于存放该 native 本地方法的局部变量表、操作数栈、动态链接、方法出口信息。方法执行完毕后,相应的栈帧也会出栈并释放内存空间。也会出现StackOverFlowErrorOutOfMemoryError 两种错误。

五、堆

        Heap 堆区,用于存放对象实例数组的内存区域。

        Heap 堆区,是 JVM所管理的内存中最大的一块区域,被所有线程共享的块内存区域。堆区中存放对象实例,“几乎”所有的对象实例以及数组都在这里分配内存。

1. 堆区的组成:新生代+老生代

        从垃圾回收的角度,由于现在收集器基本都采用分代垃圾收集思想,所以JVM 中的堆区往往进行分代划分,例如:新生代和老年代。目的是更好地回收内存,或者更快地分配内存

        堆区的组成分为新生代(Young Generation)、老年代(Old Generation)。新生代被分为伊甸区(Eden)和幸存者区(from+to),幸存区又被分为Survivor 0(from)和Survivor 1(to)。

        新生代和老年代比例为1:2,伊甸区和s0、S1比例为 8:1:1,不同区域存放对象的用途和方式不同:
        
        (1)伊甸区( Eden):存放大部分新创建对象。

        (2)幸存区(Survivor):存放 Minor Gc 之后,Eden 区和幸存区( Survivor)本身没有被回收的对象。
        
        (3)老年代:存放 Minor Gc 之后且年龄计数器达到 15 依然存活的对象、 Major GC 和 Full GC 之后仍然存活的对象。

        

2. 堆空间的大小设置

        堆区的内存大小是可修改的,默认情况下,初始堆内存为物理内存的 1/64,最大为物理内存的 1/4。

  • -xms:设置初始堆内存,例如:-Xms64m
  • -Xmx:设置最大堆内存,例如:-Xmx64m
  • -Xmn:设置新生代内存,例如:-Xmx32m

3. 创建对象的内存分配

        创建一个新对象,在堆中的分配内存。

        大部分情况下,对象会在Eden区生成,当den 区装填满的时候,会触发 Young Garbage collection,即 YGC 垃圾回收的时候,在 Eden 区实现清除策略,没有被引用的对象则直接回收。
依然存活的对象会被移送到 Survivor区。Survivor 区分为s0和s1 两块内存区域。每次 Gc 的时候,它们将存活的对象复制到未使用的 Survivor空间(s0 或 s1),然后将当前正在使用的空间完全清除,交换两块空间的使用状态。每次交换时,对象的年龄会加 +1。
        
        如果 YGC 要移送的对象大于 Survivor 区容量的上限,则直接移交给老年一个对象也不可能永远呆在新生代,在中 一个对象从新生代晋升到老年代的阈值默认值是 15,可以在 Survivor 区交换 14 次之后,晋升至老年代。

        

4. 堆区产生的错误

        堆区最容易出现的就是 OutOfMemoryError 错误,这种错误的表现形式会
有以下两种:

        (1)OutOfMemoryErrorGc overhead Limit Exceeded:当JVM 花太多
时间执行垃圾回收,并且只能回收很少的堆空间时,就会发生此错误。
        
        (2)OutofMemoryErrorJava heap space:假如在创建新的对象时,堆内存
中的空间不足以存放新创建的对象,就会引发此错误。

六、字符串常量

1. String 的两种创建方式

  • 第一种方式是在常量池中获取字符串对象
  • 第二种方式是直接在堆内存空间创建一个新的字符串对象

2. String 的intern()方法:

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

3. String的拼接

        String str1 = "str";String str2 = "ing";String str3 ="str"+"ing"; // 常量池中的新字符串对象String str4= str1 + str2; //在堆中创建的新字符串对象String str5 ="string"; // 常量池中的已有字符串对象System.out.println(str3 == str4); //falseSystem.out.println(str3 == str5); //trueSystem.out.println(str4 == str5); //false

4. String s1 = new string("abc");这句代码创建了几个字符串对象?

        创建 1或 2 个字符串。如果常量池中已存在字符串常量“abc ”,则只会在堆空间创建一个字符串常量“ abc ”。如果常量池中没有字符串常量“ abc”,那么它将首先在池中创建,然后在堆空间中创建,因此将创建总共2个字符串对象。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/1544945.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

Redis: 特点,优势,与其他产品的区别以及高并发原理

入门Redis概述 1 )选择Redis是因为其高性能 因为 Redis 它数据存储的机制是存在内存中的,减少了传统关系数据库的磁盘IO它是单线程的保证了原子性,它还提供了事务,锁等相关的机制 2 )Redis 环境安装配置 linux 或 d…

【Python-GUI图形化界面-PyQt5模块(3)】——Qwidget核心模块

本文旨在带大家学习Python中的一种GUI图形化界面模块——PyQt5模块,将为大家详细了解PyQt5模块中函数的参数和使用: 一、PyQt5简介 PyQt是Qt框架的Python语言实现,由Riverbank Computing开发,是最强大的GUI库之一。 官方网站&a…

Qt-QSpinBox输入类控件(32)

目录 描述 属性 信号 使用 描述 微调框,如下,运行用户进行细微数据的操作,点击按钮,数据就会发生 “微调” 属性 value存储的数值.singleStep每次调整的"步⻓".按下⼀次按钮数据变化多少.displayInteger数字的进制…

云服务器是干什么的?

随着云计算的发展,云服务器的功能逐步完善。但是还有不少用户不清楚云服务器是干什么的?云服务器提供了一种灵活、可扩展的计算解决方案,适用于各种在线业务和项目。提供虚拟化的计算资源是云服务器最基本也是最重要的功能。 云服务器是干什…

leetcode第169题:多数元素

给定一个大小为 n 的数组 nums ,返回其中的多数元素。多数元素是指在数组中出现次数 大于 ⌊ n/2 ⌋ 的元素。 你可以假设数组是非空的,并且给定的数组总是存在多数元素。 示例 1: 输入:nums [3,2,3] 输出:3 示例 …

内置函数sorted()与方法sort()的区别、内置函数reversed()与方法reverse()的区别

1、内置函数sorted()与方法sort() #内置函数sorted()与方法sort()的区别 #定义一个列表ls ls[4,3,6,7,9] print(sorted(ls)) print(ls)#sorted函数不会改变原列表的顺序,它只是生成了一个新列表(临时排序,不会改变与列表顺序) pr…

ARM单片机的内存分布(重要)

ARM单片机的内存分布(重要) 一、S32K344的内存布局 MEMORY {int_pflash : ORIGIN 0x00400000, LENGTH 0x003D4000 /* 4096KB - 176KB (sBAF HSE)*/int_dflash : ORIGIN 0x10000000, LENGTH 0x00020000 /* 128KB …

MySQL 缓冲池管理与常见优化技巧

在 MySQL 数据库的性能优化中,缓冲池的管理至关重要。同时,了解其他常见的优化技巧也能极大地提升数据库的运行效率。今天,我们就来深入探讨在 MySQL 中如何管理并调整缓冲池的大小,以及一些常见的优化技巧。 一、缓冲池的重要性…

关于 NLP 应用方向与深度训练的核心流程

文章目录 主流应用方向核心流程(5步)1.选定语言模型结构2.收集标注数据3.forward 正向传播4.backward 反向传播5.使用模型预测真实场景 主流应用方向 文本分类文本匹配序列标注生成式任务 核心流程(5步) 基本流程实现的先后顺序…

harmonyOS ArkTS最新跳转Navigation

文章目录 取消标题栏初始页面(load)设置为竖屏 自定义标题Tabs&TabContentTabs通过divider实现了分割线各种属性 图片下载 官方文档 Entry Component struct Index {State message: string Hello WorldState djs:number 5build() {Column(){Navigation(){}.title("g…

从0到1搭建权限管理系统系列三 .net8 JWT创建Token并使用

创建Token 创建token的因素(条件)有很多,在该篇文章中,采用jwt配置和用户基本信息作为生成token的基本因素(读者可根据系统,自由改变生成token因素)。 在JwtPlugInUnit.CS中创建2个方法&#xf…

大模型常见面试题汇总(含答案),非常详细收藏我这一篇就够了

最近秋招正在如火如荼地进行中,看到很多人的简历上都包含大模型相关的工作,各家大厂和初创都很舍得给钱,动辄百万年包也变得不再稀奇。 因此在大模型纵横的这个时代,不仅大模型技术越来越卷,就连大模型相关的岗位和面…

USB 电缆中的信号线 DP、DM 的缩写由来

经常在一些芯片的规格书中看到 USB 的信号对是以 DP 和 DM 命名: 我在想,这些规格书是不是写错了,把 N 写成 M 了?DM 中的 M 到底是什么的缩写? 于是我找了一些资料,终于在《Universal Serial Bus Cables …

‘艾’公益——微笑行动「毕节站」为艾祝福,让笑起舞

艾多美“微笑行动”毕节站拉开帷幕 此次爱心帮助77名唇腭裂患儿 重新绽放微笑 不让笑容留有缺憾 每个孩子都有微笑的权利 艾多美向唇腭裂儿童伸出援手 绽放笑容,拥抱全新的未来 2024年9月18日-9月23日,毕节市妇幼保健院迎来了艾多美--微笑行动项目…

MES系统如何集成到ERP系统里

MES系统(制造执行系统)集成到ERP系统(企业资源计划)里是一个复杂但至关重要的过程,它有助于企业实现生产计划、物料追踪、质量控制和数据分析的无缝协作,从而提高生产效率和产品质量。以下是MES系统集成到E…

8086的指令系统

今天上午综测答辩结束,感觉就很一般,但是我昨晚也操心到觉都没睡好,今天中午舍友玩P5吵得我也没睡着,感觉脑袋昏昏沉沉,汇编上课没认真听讲,晚上来补一补。还是采用GPT来讲解(水文字&#xff09…

显示屏显示缺陷检测系统源码分享

显示屏显示缺陷检测检测系统源码分享 [一条龙教学YOLOV8标注好的数据集一键训练_70全套改进创新点发刊_Web前端展示] 1.研究背景与意义 项目参考AAAI Association for the Advancement of Artificial Intelligence 项目来源AACV Association for the Advancement of Comput…

k8s前置准备:配置虚拟机网络

目录 前言查看本机ip地址修改虚拟机配置修改linux配置配置其余linux机器的网络参考文献 前言 本文的最终目的是使虚拟机内可以访问互联网,虚拟机之间可以互相访问。 虚拟机使用的是vmware,环境是windows,虚拟镜像是linux系统。 使用桥接模式…

企业微信VS钉钉:高效办公工具推荐!

这两者各有千秋,适合不同的办公场景。企业微信的优势在于与微信的紧密集成,便于与客户沟通,适合需要频繁与外部联系的企业。它提供了基本的办公自动化功能,如团队协作、审批、日程等。 钉钉则在企业管理和团队协作方面功能更全面…

Snubber电路设计

思路总结: 1.根据测试和推算得出FRA(震荡频率),进而推算出Cp(寄生电容),再根据LRC关系式推算出LP和CP,后续的Csn(吸收电容)和Rsn(吸收电阻)。得出初步的参数然后再PCBA上进行微调就可以实现通用Snub电路的设计。