终于我用JOL打破了你对java对象的所有想象

2024-02-24 16:38

本文主要是介绍终于我用JOL打破了你对java对象的所有想象,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 简介
  • JOL简介
  • 使用JOL分析VM信息
  • 使用JOL分析String
  • 使用JOL分析数组
  • 使用JOL分析自动装箱
  • 使用JOL分析引用关系
  • 总结

简介

使用面向对象的编程语言的好处就是,虽然没有女朋友,但是仍然可以new对象出来。Java是面向对象的编程语言,我们天天都在使用java来new对象,但估计很少有人知道new出来的对象到底长的什么样子,是美是丑到底符不符合我们的要去?

对于普通的java程序员来说,可能从来没有考虑过java中对象的问题,不懂这些也可以写好代码。

但是对于一个有钻研精神的极客来说,肯定会想多一些,再多一些,java中的对象到底是什么样的。

今天,小F给大家介绍一款工具JOL,可以满足大家对java对象的所有想象。

更多精彩内容且看:

  • 区块链从入门到放弃系列教程-涵盖密码学,超级账本,以太坊,Libra,比特币等持续更新
  • Spring Boot 2.X系列教程:七天从无到有掌握Spring Boot-持续更新
  • Spring 5.X系列教程:满足你对Spring5的一切想象-持续更新
  • java程序员从小工到专家成神之路(2020版)-持续更新中,附详细文章教程

更多内容请访问www.flydean.com

JOL简介

JOL的全称是Java Object Layout。是一个用来分析JVM中Object布局的小工具。包括Object在内存中的占用情况,实例对象的引用情况等等。

JOL可以在代码中使用,也可以独立的以命令行中运行。命令行的我这里就不具体介绍了,今天主要讲解怎么在代码中使用JOL。

使用JOL需要添加maven依赖:

<dependency><groupId>org.openjdk.jol</groupId><artifactId>jol-core</artifactId><version>0.10</version>
</dependency>

添加完依赖,我们就可以使用了。

使用JOL分析VM信息

首先我们看下怎么使用JOL来分析JVM的信息,代码非常非常简单:

log.info("{}", VM.current().details());

输出结果:

# Running 64-bit HotSpot VM.
# Using compressed oop with 3-bit shift.
# Using compressed klass with 3-bit shift.
# WARNING | Compressed references base/shifts are guessed by the experiment!
# WARNING | Therefore, computed addresses are just guesses, and ARE NOT RELIABLE.
# WARNING | Make sure to attach Serviceability Agent to get the reliable addresses.
# Objects are 8 bytes aligned.
# Field sizes by type: 4, 1, 1, 2, 2, 4, 4, 8, 8 [bytes]
# Array element sizes: 4, 1, 1, 2, 2, 4, 4, 8, 8 [bytes]

上面的输出中,我们可以看到:Objects are 8 bytes aligned,这意味着所有的对象分配的字节都是8的整数倍。

使用JOL分析String

上面的都不是重点,重点是怎么使用JOL来分成class和Instance信息。

其实java中的对象,除了数组,其他对象的大小应该都是固定的。我们先举一个最最常用的字符串来看一下:

log.info("{}",ClassLayout.parseClass(String.class).toPrintable());

上面的例子中,我们使用ClassLayout来解析一个String类,先看下输出:

[main] INFO com.flydean.JolUsage - java.lang.String object internals:OFFSET  SIZE      TYPE DESCRIPTION                               VALUE0    12           (object header)                           N/A12     4    byte[] String.value                              N/A16     4       int String.hash                               N/A20     1      byte String.coder                              N/A21     1   boolean String.hashIsZero                         N/A22     2           (loss due to the next object alignment)
Instance size: 24 bytes
Space losses: 0 bytes internal + 2 bytes external = 2 bytes total

先解释下各个字段的含义,OFFSET是偏移量,也就是到这个字段位置所占用的byte数,SIZE是后面类型的大小,TYPE是Class中定义的类型,DESCRIPTION是类型的描述,VALUE是TYPE在内存中的值。

分析下上面的输出,我们可以得出,String类中占用空间的有5部分,第一部分是对象头,占12个字节,第二部分是byte数组,占用4个字节,第三部分是int表示的hash值,占4个字节,第四部分是byte表示的coder,占1个字节,最后一个是boolean表示的hashIsZero,占1个字节,总共22个字节。但是JVM中对象内存的分配必须是8字节的整数倍,所以要补全2字节,最后String类的总大小是24字节。

有人可能要问小F了,如果字符串里面存了很多很多数据,那么对象的大小还是24字节吗?

这个问题问得非常有水平,下面我们就来看看怎么使用JOL来解析String对象的信息:

log.info("{}",ClassLayout.parseInstance("www.flydean.com").toPrintable());

上面的例子,我们使用了parseInstance而不是parseClass来解析String实例的信息。

输出结果:

[main] INFO com.flydean.JolUsage - java.lang.String object internals:OFFSET  SIZE      TYPE DESCRIPTION                               VALUE0     4           (object header)                           01 c2 63 a2 (00000001 11000010 01100011 10100010) (-1570520575)4     4           (object header)                           0c 00 00 00 (00001100 00000000 00000000 00000000) (12)8     4           (object header)                           77 1a 06 00 (01110111 00011010 00000110 00000000) (399991)12     4    byte[] String.value                              [119, 119, 119, 46, 102, 108, 121, 100, 101, 97, 110, 46, 99, 111, 109]16     4       int String.hash                               020     1      byte String.coder                              021     1   boolean String.hashIsZero                         false22     2           (loss due to the next object alignment)
Instance size: 24 bytes
Space losses: 0 bytes internal + 2 bytes external = 2 bytes total

先看结论,和String Class一样,这个String对象确实只占24字节。

实例的解析和Class解析的结果差不多,因为是实例对象,所以多了VALUE的值。

我们知道在JDK9之后,String的底层存储从Char[] 变成了Byte[]用于节约String的存储空间。上面的输出中,我们可以看到String.value值确实很长,但是保存在String中的只是Byte数组的引用地址,所以4字节就够了。

使用JOL分析数组

虽然String的大小是不变的,但是其底层数组的大小是可变的。我们再举个例子:

log.info("{}",ClassLayout.parseClass(byte[].class).toPrintable());

输出结果:

[main] INFO com.flydean.JolUsage - [B object internals:OFFSET  SIZE   TYPE DESCRIPTION                               VALUE0    16        (object header)                           N/A16     0   byte [B.<elements>                             N/A
Instance size: 16 bytes
Space losses: 0 bytes internal + 0 bytes external = 0 bytes total

类的解析结果,可以看到Byte数组占16个字节。

再看实例的情况:

log.info("{}",ClassLayout.parseInstance("www.flydean.com".getBytes()).toPrintable());

输出结果:

[main] INFO com.flydean.JolUsage - [B 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)                           22 13 07 00 (00100010 00010011 00000111 00000000) (463650)12     4        (object header)                           0f 00 00 00 (00001111 00000000 00000000 00000000) (15)16    15   byte [B.<elements>                             N/A31     1        (loss due to the next object alignment)
Instance size: 32 bytes
Space losses: 0 bytes internal + 1 bytes external = 1 bytes total

可以看到数组的大小真的变化了,这次变成了32字节。

使用JOL分析自动装箱

我们知道,java中的基本类型都有一个和它对于的Object类型,比如long和Long,下面我们来分析下他们两个在JVM中的内存区别:

log.info("{}",ClassLayout.parseClass(Long.class).toPrintable());

输出结果:

[main] INFO com.flydean.JolUsage - java.lang.Long object internals:OFFSET  SIZE   TYPE DESCRIPTION                               VALUE0    12        (object header)                           N/A12     4        (alignment/padding gap)                  16     8   long Long.value                                N/A
Instance size: 24 bytes
Space losses: 4 bytes internal + 0 bytes external = 4 bytes total

可以看到1个Long对象是占24个字节的,但是其中真正存储long的value只占8个字节。

看一个实例:

log.info("{}",ClassLayout.parseInstance(1234567890111112L).toPrintable());

输出结果:

[main] INFO com.flydean.JolUsage - java.lang.Long 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)                           9a 15 00 00 (10011010 00010101 00000000 00000000) (5530)12     4        (alignment/padding gap)                  16     8   long Long.value                                1234567890111112
Instance size: 24 bytes
Space losses: 4 bytes internal + 0 bytes external = 4 bytes total

使用JOL分析引用关系

上面我们使用JOL分析的是class内部的空间使用情况,那么如果有外部引用可不可以分析呢?

HashMap hashMap= new HashMap();
hashMap.put("flydean","www.flydean.com");
log.info("{}", GraphLayout.parseInstance(hashMap).toPrintable());

上面我们使用一个不同的layout:GraphLayout,它可以用来分析外部引用情况。

输出结果:

[main] INFO com.flydean.JolUsage - java.util.HashMap@57d5872cd object externals:ADDRESS       SIZE TYPE                      PATH                           VALUE7875f9028         48 java.util.HashMap                                        (object)7875f9058         24 java.lang.String          .table[14].key                 (object)7875f9070         24 [B                        .table[14].key.value           [102, 108, 121, 100, 101, 97, 110]7875f9088         24 java.lang.String          .table[14].value               (object)7875f90a0         32 [B                        .table[14].value.value         [119, 119, 119, 46, 102, 108, 121, 100, 101, 97, 110, 46, 99, 111, 109]7875f90c0         80 [Ljava.util.HashMap$Node; .table                         [null, null, null, null, null, null, null, null, null, null, null, null, null, null, (object), null]7875f9110         32 java.util.HashMap$Node    .table[14]                     (object)

从结果我们可以看到HashMap本身是占用48字节的,它里面又引用了占用24字节的key和value。

总结

使用JOL可以分析java类和对象,这个对于我们对JVM和java源代码的理解和实现都是非常有帮助的。

本文的例子https://github.com/ddean2009/
learn-java-base-9-to-20

本文作者:flydean程序那些事

本文链接:http://www.flydean.com/java-object-layout-jol/

本文来源:flydean的博客

欢迎关注我的公众号:程序那些事,更多精彩等着您!

这篇关于终于我用JOL打破了你对java对象的所有想象的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/742714

相关文章

Spring Boot 实现 IP 限流的原理、实践与利弊解析

《SpringBoot实现IP限流的原理、实践与利弊解析》在SpringBoot中实现IP限流是一种简单而有效的方式来保障系统的稳定性和可用性,本文给大家介绍SpringBoot实现IP限... 目录一、引言二、IP 限流原理2.1 令牌桶算法2.2 漏桶算法三、使用场景3.1 防止恶意攻击3.2 控制资源

Mac系统下卸载JAVA和JDK的步骤

《Mac系统下卸载JAVA和JDK的步骤》JDK是Java语言的软件开发工具包,它提供了开发和运行Java应用程序所需的工具、库和资源,:本文主要介绍Mac系统下卸载JAVA和JDK的相关资料,需... 目录1. 卸载系统自带的 Java 版本检查当前 Java 版本通过命令卸载系统 Java2. 卸载自定

springboot下载接口限速功能实现

《springboot下载接口限速功能实现》通过Redis统计并发数动态调整每个用户带宽,核心逻辑为每秒读取并发送限定数据量,防止单用户占用过多资源,确保整体下载均衡且高效,本文给大家介绍spring... 目录 一、整体目标 二、涉及的主要类/方法✅ 三、核心流程图解(简化) 四、关键代码详解1️⃣ 设置

Java Spring ApplicationEvent 代码示例解析

《JavaSpringApplicationEvent代码示例解析》本文解析了Spring事件机制,涵盖核心概念(发布-订阅/观察者模式)、代码实现(事件定义、发布、监听)及高级应用(异步处理、... 目录一、Spring 事件机制核心概念1. 事件驱动架构模型2. 核心组件二、代码示例解析1. 事件定义

SpringMVC高效获取JavaBean对象指南

《SpringMVC高效获取JavaBean对象指南》SpringMVC通过数据绑定自动将请求参数映射到JavaBean,支持表单、URL及JSON数据,需用@ModelAttribute、@Requ... 目录Spring MVC 获取 JavaBean 对象指南核心机制:数据绑定实现步骤1. 定义 Ja

javax.net.ssl.SSLHandshakeException:异常原因及解决方案

《javax.net.ssl.SSLHandshakeException:异常原因及解决方案》javax.net.ssl.SSLHandshakeException是一个SSL握手异常,通常在建立SS... 目录报错原因在程序中绕过服务器的安全验证注意点最后多说一句报错原因一般出现这种问题是因为目标服务器

Python打印对象所有属性和值的方法小结

《Python打印对象所有属性和值的方法小结》在Python开发过程中,调试代码时经常需要查看对象的当前状态,也就是对象的所有属性和对应的值,然而,Python并没有像PHP的print_r那样直接提... 目录python中打印对象所有属性和值的方法实现步骤1. 使用vars()和pprint()2. 使

Java实现删除文件中的指定内容

《Java实现删除文件中的指定内容》在日常开发中,经常需要对文本文件进行批量处理,其中,删除文件中指定内容是最常见的需求之一,下面我们就来看看如何使用java实现删除文件中的指定内容吧... 目录1. 项目背景详细介绍2. 项目需求详细介绍2.1 功能需求2.2 非功能需求3. 相关技术详细介绍3.1 Ja

springboot项目中整合高德地图的实践

《springboot项目中整合高德地图的实践》:本文主要介绍springboot项目中整合高德地图的实践,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一:高德开放平台的使用二:创建数据库(我是用的是mysql)三:Springboot所需的依赖(根据你的需求再

spring中的ImportSelector接口示例详解

《spring中的ImportSelector接口示例详解》Spring的ImportSelector接口用于动态选择配置类,实现条件化和模块化配置,关键方法selectImports根据注解信息返回... 目录一、核心作用二、关键方法三、扩展功能四、使用示例五、工作原理六、应用场景七、自定义实现Impor