文章目录

  • Pre
  • 指针压缩
    • 论证压缩效果
    • UseCompressedOops & UseCompressedClassPointers
    • 【指针压缩】开启 VS 关闭
    • 指针压缩的目的
    • 为什么堆内存最好不要超过32G

Pre

JVM - 剖析Java对象头Object Header之对象大小

mark word : 32位 占4字节 ,64位 占 8字节

klass point : 开启压缩占4字节,未开启 占 8字节。


指针压缩

论证压缩效果

  • jdk1.6 update14开始,在64bit操作系统中,JVM支持指针压缩
  • 启用指针压缩-XX:+UseCompressedOops(默认开启),禁止指针压缩:-XX:-UseCompressedOops

oop(ordinary object pointer) 就是对象指针的意思。

运行参数增加

 -XX:-UseCompressedOops

禁用指针压缩,我们来看下对象头的大小

package com.gof.test;import org.openjdk.jol.info.ClassLayout;/*** @author 小工匠* @version v1.0* @create 2020-06-25 16:21* @motto show me the code ,change the word* @blog https://artisan.blog.csdn.net/* @description**/public class ObjectHeaderTest {public static void main(String[] args) {ClassLayout layout = ClassLayout.parseInstance(new Object());System.out.println(layout.toPrintable());System.out.println();ClassLayout layout1 = ClassLayout.parseInstance(new int[]{});System.out.println(layout1.toPrintable());System.out.println();ClassLayout layout2 = ClassLayout.parseInstance(new ArtisanTest());System.out.println(layout2.toPrintable());}// -XX:+UseCompressedOops           默认开启的压缩所有指针// -XX:+UseCompressedClassPointers  默认开启的压缩对象头里的类型指针Klass Pointer// Oops : Ordinary Object Pointerspublic static class ArtisanTest {//8B mark word//4B Klass Pointer   如果关闭压缩-XX:-UseCompressedClassPointers或-XX:-UseCompressedOops,则占用8Bint id;        //4BString name;   //4B  如果关闭压缩-XX:-UseCompressedOops,则占用8Bbyte b;        //1BObject o;      //4B  如果关闭压缩-XX:-UseCompressedOops,则占用8B}
}

【输出结果】

java.lang.Object object internals:OFFSET  SIZE   TYPE DESCRIPTION                               VALUE0     4        (object header)                           01 00 00 00 (00000001 00000000 00000000 00000000) (1)4     4        (object header)                           00 00 00 00 (00000000 00000000 00000000 00000000) (0)8     4        (object header)                           00 1c e4 17 (00000000 00011100 11100100 00010111) (400825344)12     4        (object header)                           00 00 00 00 (00000000 00000000 00000000 00000000) (0)
Instance size: 16 bytes
Space losses: 0 bytes internal + 0 bytes external = 0 bytes total[I object internals:OFFSET  SIZE   TYPE DESCRIPTION                               VALUE0     4        (object header)                           01 00 00 00 (00000001 00000000 00000000 00000000) (1)4     4        (object header)                           00 00 00 00 (00000000 00000000 00000000 00000000) (0)8     4        (object header)                           68 0b e4 17 (01101000 00001011 11100100 00010111) (400821096)12     4        (object header)                           00 00 00 00 (00000000 00000000 00000000 00000000) (0)16     4        (object header)                           00 00 00 00 (00000000 00000000 00000000 00000000) (0)20     4        (alignment/padding gap)                  24     0    int [I.<elements>                             N/A
Instance size: 24 bytes
Space losses: 4 bytes internal + 0 bytes external = 4 bytes totalcom.gof.test.ObjectHeaderTest$ArtisanTest object internals:OFFSET  SIZE               TYPE DESCRIPTION                               VALUE0     4                    (object header)                           05 00 00 00 (00000101 00000000 00000000 00000000) (5)4     4                    (object header)                           00 00 00 00 (00000000 00000000 00000000 00000000) (0)8     4                    (object header)                           f8 a5 4e 18 (11111000 10100101 01001110 00011000) (407807480)12     4                    (object header)                           00 00 00 00 (00000000 00000000 00000000 00000000) (0)16     4                int ArtisanTest.id                            020     1               byte ArtisanTest.b                             021     3                    (alignment/padding gap)                  24     8   java.lang.String ArtisanTest.name                          null32     8   java.lang.Object ArtisanTest.o                             null
Instance size: 40 bytes
Space losses: 3 bytes internal + 0 bytes external = 3 bytes total

我们先把默认的开启指针压缩的这个的测试结果题图,方便比对

【默认开启指针压缩】

VS

【关闭指针压缩】


【默认开启指针压缩】

VS

【关闭指针压缩】


最后一个,对于包含多个变量的对象的对象头

【默认开启指针压缩】

VS

【关闭指针压缩】


UseCompressedOops & UseCompressedClassPointers

  -XX:+UseCompressedOops           默认开启的压缩所有指针-XX:+UseCompressedClassPointers  默认开启的压缩对象头里的类型指针Klass Pointer

【指针压缩】开启 VS 关闭

类型 开启指针压缩 关闭指针压缩
Object 16 16
int数组 16 24
ArtisanTest对象 32 40

指针压缩的目的

同一个对象, 不开启指针压缩 8字节 存入堆中和 开启指针压缩4字节存入堆中,哪个更好一些,显而易见。

简言之:为了更好地节省内存,避免GC压力过大。

同时在64位平台的HotSpot中使用32位指针(实际存储用64位),内存使用会多出1.5倍左右,使用较大指针在主内存和缓存之间移动数据,占用较大宽带。

所以为了减少64位平台下内存的消耗,JVM在1.6以后默认启用指针压缩功能。


为什么堆内存最好不要超过32G

在jvm中,32位地址最大支持4G内存(2的32次方) 。

我们知道以前32位的操作系统 ,内存格中最多存放32位 ,所以
2的32次方 等于4294967296 字节 = 4 G 。

64位,不是2的64次方,这个值简直太大了。。。。

在jvm中,32位地址最大支持4G内存(2的32次方),可以通过对对象指针的存入堆内存时压缩编码、取出到cpu寄存器后解码方式进行优化

举个例子 对象指针在堆中是32位,在寄存器中是35位,2的35次方=32G),使得jvm只用32位地址就可以支持更大的内存配置(小于等于32G) 。


JVM如何处理的?

  • 当堆内存小于4G时,不需要启用指针压缩,jvm会直接去除高32位地址,即使用低虚拟地址空间

  • 当堆内存大于32G时,压缩指针会失效,会强制使用64位(即8字节)来对java对象寻址, 那这样的话内存占用较大,GC压力等等

JVM - 剖析Java对象头Object Header之指针压缩相关推荐

  1. JVM - 剖析Java对象头Object Header之对象大小

    文章目录 Pre 总览 对象头剖析 查看对象内存的占用情况 对象头C++源码 注释 Pre JVM - 写了这么多年代码,你知不道new对象背后的逻辑? 中大体介绍了Java中 new 对象背后的主要 ...

  2. 64位JVM的Java对象头详解

    关注"Java艺术"一起来充电吧! 我们编写一个Java类,编译后会生成.class文件,当类加载器将class文件加载到jvm时,会生成一个Klass类型的对象(c++),称为类 ...

  3. java的头怎么写_JAVA对象布局之对象头(Object Header)

    由于Java面向对象的思想,在JVM中需要大量存储对象,存储时为了实现一些额外的功能,需要在对象中添加一些标记字段用于增强对象功能 .在学习并发编程知识synchronized时,我们总是难以理解其实 ...

  4. JAVA对象布局之对象头(Object Header)

    由于Java面向对象的思想,在JVM中需要大量存储对象,存储时为了实现一些额外的功能,需要在对象中添加一些标记字段用于增强对象功能 .在学习并发编程知识synchronized时,我们总是难以理解其实 ...

  5. 【Java】Java 对象头 真的是 32位 吗?

    1.概述 HotSpot虚拟机中,对象在内存中存储的布局可以分为三块区域: 对象头(Header) 实例数据(Instance Data) 对齐填充(Padding). 是真的吗?假的! 假的!!! ...

  6. java对象头_浅谈java对象结构 对象头 Markword

    概述 对象实例由对象头.实例数据组成,其中对象头包括markword和类型指针,如果是数组,还包括数组长度; | 类型 | 32位JVM | 64位JVM| | ------ ---- | ----- ...

  7. 盘一盘 synchronized (一)—— 从打印Java对象头说起

    Java对象头的组成 Java对象的对象头由 mark word 和  klass pointer 两部分组成, mark word存储了同步状态.标识.hashcode.GC状态等等. klass  ...

  8. java对象头markword_浅谈java对象结构 对象头 Markword

    概述 对象实例由对象头.实例数据组成,其中对象头包括markword和类型指针,如果是数组,还包括数组长度; | 类型 | 32位JVM | 64位JVM| | ------ ---- | ----- ...

  9. java打印对象头信息_打印Java对象头

    打印Java对象头 对象头形式 JVM中对象头的方式有以下两种(以32位JVM为例)普通对象|----------------------------------------------------- ...

最新文章

  1. 新手C#string类常用函数的学习2018.08.04
  2. 移动端调试利器------微信开源项目
  3. session_start() 对 HTTP_REQUEST扩展/fsockopen函数 的影响
  4. U盘:对于目标文件系统,文件过大放不到u盘里
  5. 微服务四个常见问题,以及SpringCloud Netflix和SpringCloud Alibaba和Apache Dubbo zookeeper区别
  6. CLR via C# 边读边想 03 - 本地程序集和强命名程序集
  7. 粗钢行业调研报告 - 市场现状分析与发展前景预测(2021-2027年)
  8. (一)PyQt5系列教程:使用PyQt5创建一个简单的demo
  9. ColorPix最好用的屏幕取色器
  10. Hexo NexT主题更改语言
  11. 安卓手机管理器_手机QQ聊天记录如何导出到电脑上查看并打印
  12. PDF转高清图片怎么转?推荐这款PDF转高清图片软件!
  13. Ansys Meshing
  14. spark进行数据清洗时,如何读取xlsx表格类型文件
  15. 前端,java后端开发,数据分析师应该掌握的技术,不要盲目跟风
  16. NTC,PT100热电阻转4-20mA温度信号转换器
  17. WEB前端工程师 – 职业生涯规划
  18. Mybatis xml映射文件错误,导致Tomcat无法启动,也不报异常
  19. Win10安装MySQL8压缩包版
  20. mysql8主从复制集群_mysql8主从复制服务器搭建

热门文章

  1. 读取.bin激光雷达点云文件格式并可视化
  2. C++继承时的名字遮蔽(二)
  3. opencv 车牌切割
  4. c语言数据储存系统,编的学生成绩管理系统 从文件中读取保存数据总会多读入一组乱码数据...
  5. 整数的二进制表达中有多少个1
  6. 将搜索二叉树转换成双向链表
  7. 文巾解题 45. 跳跃游戏 II
  8. 【数学建模】MATLAB应用实战系列(九十二)-教你怎么挑对象,层次分析法应用案例(附MATLAB代码)
  9. 小白入门PyTorch | 第一篇:什么是PyTorch?
  10. OSM OpenStreetMap 获取城市路网数据及转为ESRI shp数据的方法