OpenJDK 15+ 内存优化:深入理解大堆场景下的压缩类指针

OpenJDK 15+ 内存优化:深入理解大堆场景下的压缩类指针

Java 15及更高版本在处理超过32GB的大堆内存时,通过独立压缩类指针(Compressed Class Pointers)显著优化了对象的内存占用。这一改进使得即使对象引用本身无法压缩,对象的元数据开销也能保持较低水平,从而提升了内存效率,解决了早期Java版本中压缩类指针与压缩对象指针绑定导致的内存膨胀问题。本文将深入探讨这一机制,解释为何在OpenJDK 19中,即使面对大堆内存,Java对象的内存占用依然能够保持较低水平。

理解Java中的指针压缩机制

java虚拟机(jvm)中,为了节省内存并提高缓存效率,引入了指针压缩(compressed pointers)机制。它主要分为两类:

压缩对象指针 (Compressed Ordinary Object Pointers, Compressed Oops):当Java堆内存大小在一定范围内(通常是32GB以下)时,JVM可以将64位对象引用(8字节)压缩为32位(4字节)。这是通过将对象地址视为相对于堆基址的偏移量,并假设对象总是8字节对齐来实现的。这样,一个32位的偏移量可以表示高达2^35字节(32GB)的地址空间。当堆内存超过32GB时,Compressed Oops 通常会自动禁用,对象引用会恢复为8字节。

压缩类指针 (Compressed Class Pointers):每个Java对象在内存中都包含一个指向其Class对象的指针。这个指针指向了该对象的类型信息,是对象头的一部分。与Compressed Oops类似,Compressed Class Pointers 机制旨在将这个Class指针从8字节压缩为4字节,从而进一步减少每个对象的内存开销。

Java 11与Java 19大堆内存表现差异的根源

用户可能会观察到,在Java 11中,当堆内存超过32GB时,对象的内存占用会显著增加,而在Java 19中,即使堆内存达到41GB,某些对象的内存占用却依然较低,这似乎与Compressed Oops的32GB限制相矛盾。这种差异的根源不在于对象引用(Oops)本身的压缩,而在于类指针(Class Pointer)的压缩行为

在JDK 15之前,UseCompressedClassPointers 这个JVM选项与 UseCompressedOops 选项是紧密绑定的。这意味着:

当堆内存小于32GB,UseCompressedOops 默认启用时,UseCompressedClassPointers 也会随之启用,类指针被压缩为4字节。当堆内存大于32GB,UseCompressedOops 自动禁用时,UseCompressedClassPointers 也会被隐式禁用,导致类指针恢复为8字节。

因此,在JDK 11(早于JDK 15)中,当堆内存设置为41GB时,不仅对象引用会是8字节,每个对象的类指针也会是8字节,从而增加了对象的整体内存开销。

JDK 15及之后的改进:类指针的独立压缩

OpenJDK社区通过 JDK-8241825: Make compressed oops and compressed class pointers independent 这一改进,解决了上述绑定问题。自JDK 15起,UseCompressedClassPointers 选项变得独立于 UseCompressedOops。这意味着:

即使堆内存超过32GB,导致 UseCompressedOops 禁用(对象引用为8字节),UseCompressedClassPointers 仍然可以保持启用状态。因此,即使在大堆场景下,对象的类指针依然可以被压缩为4字节。

这个改变显著降低了每个Java对象的元数据开销,即使在无法压缩对象引用的大堆内存环境中,也能实现内存效率的提升。

示例分析:验证类指针的压缩效果

为了直观地展示这一变化,我们可以使用 Java Object Layout (JOL) 工具来分析对象的内存布局。以下代码片段展示了如何打印一个 Object 数组的内存布局:

import org.openjdk.jol.info.ClassLayout;public class ObjectMemoryLayout {    public static void main(String[] args) {        // 打印一个包含3个Object引用的数组的内存布局        System.out.println(ClassLayout.parseInstance(new Object[3]).toPrintable());    }}

使用上述代码,并在不同JDK版本和堆大小下运行,我们可以观察到以下差异:

1. JDK 11 (或更早版本,堆大小 41GB)

在JDK 11环境下,使用 -Xms41g -Xmx41g 运行,Object 数组的内存布局可能类似如下:

存了个图 存了个图

视频图片解析/字幕/剪辑,视频高清保存/图片源图提取

存了个图 17 查看详情 存了个图

[Ljava.lang.Object; object internals:OFF  SZ               TYPE DESCRIPTION               VALUE  0   8                    (object header: mark)     0x0000000000000001 (non-biasable; age: 0)  8   8                    (object header: class)    0x000001f54bec41e0  <-- 注意这里是8字节 16   4                    (array length)            3 20   4                    (alignment/padding gap) 24  24   java.lang.Object Object;.        N/AInstance size: 48 bytesSpace losses: 4 bytes internal + 0 bytes external = 4 bytes total

可以看到,object header: class 字段占用了8字节,导致整个 Object[3] 实例的大小为48字节。这是因为 UseCompressedClassPointers 随 UseCompressedOops 一同被禁用。

2. JDK 15 或更新版本 (例如 JDK 19,堆大小 41GB)

在JDK 15或更高版本(如JDK 19)环境下,使用 -Xms41g -Xmx41g 运行,Object 数组的内存布局可能类似如下:

[Ljava.lang.Object; object internals:OFF  SZ               TYPE DESCRIPTION               VALUE  0   8                    (object header: mark)     0x0000000000000001 (non-biasable; age: 0)  8   4                    (object header: class)    0x000020fc          <-- 注意这里是4字节 12   4                    (array length)            3 16  24   java.lang.Object Object;.        N/AInstance size: 40 bytesSpace losses: 0 bytes internal + 0 bytes external = 0 bytes total

在此输出中,object header: class 字段仅占用4字节。由于类指针的压缩,以及相应的填充调整,整个 Object[3] 实例的大小减少到了40字节。值得注意的是,Object;. 部分(即数组中存储的3个 Object 引用)在两种情况下都占用了24字节(每个引用8字节,因为 UseCompressedOops 已禁用),这进一步证明了差异在于类指针,而非对象引用本身。

结论与最佳实践

从JDK 15开始,Java虚拟机在处理大堆内存时,通过使压缩类指针独立于压缩对象指针,显著优化了对象的内存占用。这意味着即使您的应用程序需要超过32GB的堆内存,并且无法利用Compressed Oops来压缩对象引用,JVM仍然可以通过压缩类指针来减少每个对象的固定开销。

主要启示:

区分概念: 理解“压缩对象指针”(Compressed Oops)和“压缩类指针”(Compressed Class Pointers)是两个独立但相关的优化。JDK 15+ 内存优势: 对于运行在大堆内存环境下的Java应用程序,升级到JDK 15或更高版本可以带来潜在的内存效率提升,减少对象元数据开销。工具辅助分析: 使用如JOL (Java Object Layout) 这样的工具是分析和理解Java对象内存布局的强大手段,有助于开发者深入探究内存使用情况并进行优化。

通过利用这些JVM内部的优化,开发者可以更好地管理和利用系统资源,尤其是在处理大规模数据和高并发场景下的Java应用程序时。

以上就是OpenJDK 15+ 内存优化:深入理解大堆场景下的压缩类指针的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/215746.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月3日 14:44:04
下一篇 2025年11月3日 14:47:12

相关推荐

  • PHP静态页面如何与数据库交互?

    如何让 php 静态页面与数据库交互 当您学习 php 时,连接静态页面和数据库以管理数据非常重要。以下步骤将指导您完成此过程: 1. 连接到数据库 在您的 php 脚本中,使用 mysqli_connect() 函数建立与数据库的连接。 $conn = mysqli_connect(‘localh…

    2025年12月9日
    000
  • 如何从头开始为 PHP 应用程序构建路由系统

    如果您刚刚开始 PHP 开发之旅, 您很可能在 URL 中使用完整的文件名来导航 应用程序,例如 server/contact.php。不用担心,我们都是这样开始的,这就是我们学习的方式。 今天,我想帮助您改进在浏览器中导航文件的方式。 应用。我们将讨论路由,因为它在任何情况下都至关重要 现代应用。…

    2025年12月9日
    000
  • 如何用JavaScript限制单选评分元素,防止用户重复点击?

    如何限制多个评分元素,在选择一个后阻止其他元素被点击 您想要实现的功能是,在一个包含多个评分元素(例如按钮或链接)的列表中,当点击一个元素时,其他元素将被禁用且无法再次点击。 要实现此功能,可以采用以下步骤: 为 元素添加一个 id 立即学习“Java免费学习笔记(深入)”; 1 2 3 4 添加事…

    2025年12月9日
    000
  • 如何只允许用户单击一次评价选项,并阻止其他选项被点击?

    如何实现点击一个元素后,其他同类元素无法再次点击? 有四个评价选项,当用户单击其中一个评价并添加“on”类后,其他评价选项应该变为不可点击,并提示用户已经评价。 解决方法: 在 ul 标签上添加一个 id: 评价 1 评价 2 评价 3 评价 4 然后,使用 javascript 遍历 li 标签,…

    2025年12月9日
    000
  • 如何解决大小写敏感的URL跳转问题?

    如何解决大小写不敏感的内容地址跳转问题 您希望将包含小写路径(例如“http://xxxx/oa/pms/”)的地址重定向到相应的大写路径(“http://xxxx/oa/pms/”)。 解决方案 javascript 解决方案 在 index.html 页面中添加以下 javascript 代码:…

    2025年12月9日
    000
  • 甘特图选择困难症?过来人推荐哪款好用?

    最佳甘特图推荐:实践经验分享 在甘特图选择上苦苦寻求?以下是我们的推荐,专为那些亲身体验过的用户准备。 问题: 有推荐的甘特图推荐吗?最好是有实践经验的。市面上有不少选择,Ext JS 的官方网站却没有提及。是否有 Ext JS 的甘特图推荐? 回答: 经过深入搜索和实际使用,我们推荐以下甘特图: …

    2025年12月9日
    000
  • 有哪些好用的甘特图工具推荐?

    使用过的好用甘特图工具 对于甘特图工具,推荐使用 https://github.com/taitems/jQuery.Gantt/,因为它操作便捷且功能强大,许多程序员都对该工具给予了肯定的评价。 尽管 ExtJS 官网上可能没有专门的甘特图组件,但您可以使用第三方库或插件来实现甘特图功能。例如,j…

    2025年12月9日
    000
  • PHP中如何将XML文件处理结果存入变量?

    将处理xml文件的结果存入变量中 php中,你可以利用simplexml_load_string()函数将xml字符串转换为simplexml对象,该对象可以方便地访问xml文档中的数据。以下是一个示例,展示如何将xml文件的结果存入变量中: $string = <<<XML st…

    2025年12月9日
    000
  • PHP如何读取和处理XML文件并将数据保存到变量中?

    php 读取和处理 xml 文件 为了将 xml 文件中的数据保存到变量中,我们需要利用 php 中的内置函数。这通常涉及以下步骤: 加载 xml 文件: $xmlstring = ‘ status message remainpoint taskid successcounts’;$xmlobje…

    2025年12月9日
    000
  • PHP如何将XML文件内容解析并存储到变量中?

    php处理xml文件,将结果存入变量中 通过php对xml文件进行处理,可以将xml节点中的值提取出来,并存储到php变量中,以便于后期使用。具体步骤如下: 使用simplexml_load_string()函数加载xml字符串:可以使用该函数将xml字符串转换为simplexml对象,方便后续操作…

    2025年12月9日
    000
  • 如何将多层嵌套的JSON对象转换为易于操作的多维数组?

    给的json数据中,有很多层的对象,将对象嵌套多层不直观,所以一般我们将其转换为多维数组进行操作,转换方法便是遍历对象,然后将对象的每个属性挂载到该层级上,如果对象的属性具有子属性,则继续进行嵌套操作.具体操作: ‘use strict’;function convert(obj,keys,arr)…

    2025年12月9日
    000
  • Ubuntu下PHP无法创建目录或写入文件:如何解决权限问题?

    ubuntu 下 php 无法创建目录和写入文件 在 ubuntu 中配置 lamp 环境时,用户可能会遇到 php 无法创建目录及写入文件的情况。即使已将项目权限设置为 777,apache 仍会报告错误。本文将针对此问题提供解决方案。 php 代码片段如下: $max_size = 10000;…

    2025年12月9日
    000
  • 如何实现PHP AES RSA加密算法与C#和Java的互通?

    php aes rsa 算法修改以与 c# 和 java 互通 您希望将 php 中的 aes 和 rsa 加密算法修改为与 c# 和 java 互通。具体来说,您希望能够使用这些语言相互加密和解密数据。 一种实现此目标的途径是创建一个 php 加密服务。该服务将作为一个中间层,允许您使用 c# 和…

    2025年12月9日
    000
  • 如何用jQuery实现类似谷歌搜索的自动提示功能?

    实现类似google suggest的功能 问题: 如何实现类似谷歌搜索框的自动提示功能? 回答: 可以使用 [jquery ui 自动提示](http://jqueryui.com/autocomplete/) 来实现此功能。 该插件为 控件提供自动完成功能。它从预定义的选项列表中提取匹配建议,并…

    2025年12月9日
    000
  • 如何解决系统参数设置中出现的“hash_file: failed to open stream”错误?

    根据企业情况设置系统参数 在设置系统参数时,如果遇到“hash_file(up/1437616281如何根据企业情况设置系统参数?.doc): failed to open stream: Invalid argument”错误时,可以通过以下方法解决: 当输入的路径是中文路径时,系统不会报错。但是…

    2025年12月9日
    000
  • 如何从数据库中获取数据并以 PHP 形式形成?

    要从 PHP 数据库中获取数据并将其显示在表单中,通常需要执行以下步骤:1.连接到数据库:使用 MySQLi 或 PDO 建立到数据库的连接。2.查询数据库:执行SQL查询以检索所需的数据。3.获取数据:从查询结果中获取数据。4.填充表单:使用获取的数据填写表单字段。 这是一个使用的简单示例MySQ…

    2025年12月9日
    000
  • 创建专注的领域应用程序 Symfony 方法(返回结果)

    介绍 这是本系列的最后一篇文章。在上一篇文章中,我们创建了一个应用程序服务,它使用 userentitybuilder 服务来创建实体。然后,使用条令实体管理器(这是一个基础设施服务)来持久化和刷新实体。 现在,是时候将结果返回到表示层了。 我想记住,在本系列的所有文章中,我们都将学说实体视为域实体…

    2025年12月9日
    000
  • (我的第一次)安装 Laravel

    有时,尤其是当您刚刚开始职业生涯时,您似乎遵循了指示却一事无成 – 而其他人似乎发现这非常容易。 这可能非常令人沮丧,我想描述一下即使在几十年之后我也经历完全相同的事情的几种方式。所以我在这里,试图详细描述我在努力让事情顺利进行时所犯的错误和失误。这是我关于这个主题的第一篇文章,但我希望…

    2025年12月9日
    000
  • PHP 与 MySQL:终极分步指南

    php 是一种语言,可让您在开发网页时灵活地连接和使用不同的数据库。有不同的数据库,既有商业的,也有免费使用的。其中,mysql 是与 php 并列最常用的数据库。 MySQL 是一个开源、免费使用的关系型数据库管理 系统(关系数据库管理系统)。它是一个快速、简单且高度可扩展的程序 因此可用于小型和…

    2025年12月9日 好文分享
    000
  • 您需要的 PHP CRUD 操作的最佳指南

    crud 操作通常在数据库上执行,因此,在本 php crud 操作教程中,您将借助 php 在 mysql 数据库上实现 crud 技术。    crud 缩写包含在关系数据库上执行的所有主要操作。它代表: c = 创建 r = 读取 u = 更新 d = 删除 你现在就会明白不同操作的详细信息。…

    2025年12月9日 好文分享
    000

发表回复

登录后才能评论
关注微信