返回顶部
首页 > 资讯 > 后端开发 > Python >浅谈JVM中的JOL
  • 764
分享到

浅谈JVM中的JOL

2024-04-02 19:04:59 764人浏览 泡泡鱼

Python 官方文档:入门教程 => 点击学习

摘要

目录JOL简介使用JOL分析VM信息使用JOL分析String使用JOL分析数组使用JOL分析自动装箱使用JOL分析引用关系总结JOL简介 JOL的全称是Java Object La

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                               VALUE

      0    12           (object header)                           N/A

     12     4    byte[] String.value                              N/A

     16     4       int String.hash                               N/A

     20     1      byte String.coder                              N/A

     21     1   boolean String.hashIsZero                         N/A

     22     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字节。

如果字符串里面存了很多很多数据,那么对象的大小还是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                               VALUE

      0     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                               0

     20     1      byte String.coder                              0

     21     1   boolean String.hashIsZero                         false

     22     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                               VALUE

      0    16        (object header)                           N/A

     16     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                               VALUE

      0     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/A

     31     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                               VALUE

      0    12        (object header)                           N/A

     12     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                               VALUE

      0     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                           VALUE

        7875f9028         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源代码的理解和实现都是非常有帮助的。

以上就是浅谈JVM中的JOL的详细内容,更多关于JVM中的JOL的资料请关注编程网其它相关文章!

--结束END--

本文标题: 浅谈JVM中的JOL

本文链接: https://lsjlt.com/news/127419.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 浅谈JVM中的JOL
    目录JOL简介使用JOL分析VM信息使用JOL分析String使用JOL分析数组使用JOL分析自动装箱使用JOL分析引用关系总结JOL简介 JOL的全称是Java Object La...
    99+
    2024-04-02
  • 浅谈JVM系列之JIT中的Virtual Call
    目录Virtual Call和它的本质Virtual Call和classic callVirtual Call优化单实现方法的例子Virtual Call优化多实现方法的例子总结V...
    99+
    2024-04-02
  • 浅谈JDK9的特性之JVM的xlog
    目录简介xlog的使用selectionsoutputdecorators简介 JVM是java程序运行的基础,JVM中各种事件比如:GC,class loading,JPMS,he...
    99+
    2024-04-02
  • 浅谈JVM之java class文件的密码本
    目录简介一个简单的classClassFile的二进制文件class文件的密码本magicversion常量池描述符access_flagsthis_class和super_clas...
    99+
    2024-04-02
  • 浅谈一下JVM垃圾回收算法
    目录标记-清除(Mark-Sweep)算法复制(Copying)算法Appel 式回收标记-整理(Mark-Compact)算法总结Java有着自己一套的内存管理机制,不需要开发者去...
    99+
    2023-05-18
    JVM 算法 JVM垃圾回收
  • 浅谈JVM 底层解析 i++和 ++i 区别
    目录一、前言二、代码实现三、字节码指令四、字节码解析1. 第一类问题2. 第二类问题3. 第三类问题4. 第四类问题一、前言 如果只用普通的知识解释i++和++i的话 i++ 先将...
    99+
    2024-04-02
  • 浅谈JVM之类的加载链接和初始化
    目录加载运行时常量池类加载器链接验证准备解析初始化总结加载 JVM可以分为三大部分,五大空间和三大引擎,要讲起来也不是特别复杂,先看下面的总体的JVM架构图。 从上面的图中,我们可...
    99+
    2024-04-02
  • 浅谈JVM之使用JFR解决内存泄露
    目录简介一个内存泄露的例子使用JFR和JMC来分析内存泄露OldObjectSample简介 虽然java有自动化的GC,但是还会有内存泄露的情况。当然java中的内存泄露跟C++中...
    99+
    2024-04-02
  • 浅谈Java堆外内存之突破JVM枷锁
    对于有Java开发经验的朋友都知道,Java中不需要手动的申请和释放内存,JVM会自动进行垃圾回收;而使用的内存是由JVM控制的。那么,什么时机会进行垃圾回收,如何避免过度频繁的垃圾回收?如果JVM给的内存不够用,怎么办?此时,堆外内存登场...
    99+
    2023-05-30
    java jvm 堆外内存
  • 浅谈Unity中的Shader
    目录一、Shader基础知识1.1、什么是Shader1.2、OpenGL的渲染流程1.3、shader的种类1.4、shader的开发语言二、Unity中Shader知识介绍2.1...
    99+
    2024-04-02
  • 浅谈JVM系列之从汇编角度分析NullCheck
    目录一个普通的virtual call普通方法中的null check反优化的例子一个普通的virtual call 我们来分析一下在方法中调用list.add方法的例子: pu...
    99+
    2024-04-02
  • 浅谈JVM垃圾回收有哪些常用算法
    一、前言: 垃圾回收: 在未来的JDK中可能G1会为ZGC所取代 先问自己几个问题: 什么是垃圾? 垃圾就是堆内存中(范指)没有任何指针指向的对象实体。不具有可达性。 为...
    99+
    2024-04-02
  • 浅谈JVM内存溢出原因和解决思路
    目录栈溢出(虚拟机栈和本地方法栈) 产生原因解决思路堆溢出 产生原因解决思路方法区和运行时常量池溢出 产生原因解决思路本机直接内存溢出 产生原因解决思路栈溢出(虚拟机栈和本地方法栈)...
    99+
    2024-04-02
  • 浅谈python中的多态
    目录一、多态二、多态性三、鸭子类型一、多态 多态是指一类事物有多种形态,比如动物类,可以有猫,狗,猪等等。(一个抽象类有多个子类,因而多态的概念依赖于继承) import abc class Animal(met...
    99+
    2022-06-02
    python 多态
  • 浅谈Shell中的函数
    函数可以让我们将一个复杂功能划分成若干模块,让程序结构更加清晰,代码重复利用率更高。像其他编程语言一样,Shell也支持函数。Shell函数必须先定义后使用。 1.Shell函数的定义格式 可以带function关键字...
    99+
    2022-06-04
    shell 函数
  • 浅谈Vue中的this.$store.state.xx.xx
    目录Vue this.$store.state.xx.xx获取store中的数据我的项目文件结构vue项目都在什么时候用store.state、$store.state和this.$...
    99+
    2024-04-02
  • 浅谈MySQL中的group by
    目录1、前言 2、准备user表2.1 group by规则2.2 group by使用2.3 having使用2.4 order by与limit2.5 with rol...
    99+
    2024-04-02
  • 浅谈Node.js中的定时器
    Node.js中定时器的实现 上一篇博文提到,在Node中timer并不是通过新开线程来实现的,而是直接在event loop中完成。下面通过几个JavaScript的定时器示例以及Node相关源码来分析在...
    99+
    2022-06-04
    定时器 浅谈 Node
  • 浅谈python中的占位符
    占位符,顾名思义就是插在输出里站位的符号。我们可以把它理解成我们预定饭店。当我们告诉饭店的时候,饭店的系统里会有我们的预定位置。虽然我们现在没有去但是后来的顾客就排在我们后面。 常见的占位符有三种:   1...
    99+
    2022-06-04
    浅谈 python
  • 浅谈Java中的Queue家族
    目录Queue接口Queue的分类BlockingQueueDequeTransferQueueQueue接口 先看下Queue的继承关系和其中定义的方法: Queue继承自Col...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作