一文了解JVM(中)

jvm · 浏览次数 : 2

小编点评

HotSpot虚拟机对象创建流程: 1. 检查常量池中是否已经加载相应的类。 2. 如果没有,先执行类加载。 3. 分配内存。 * 如果Java堆规整,采用指针碰撞方式分配内存。 * 如果Java堆不规整,采用空闲列表方式分配内存。 4. 初始化内存空间。 5. 对象设置(元信息、哈希码等)。 6. 执行方法。 7. 为对象分配内存。 对象的访问定位方式主要有两种: 1. 句柄:指向对象的指针,维护着对象的指针。 2. 直接指针:指向对象的内存地址。 HotSpot中采用的是直接指针访问方式。 关于Java程序判断JVM是32位还是64位的方法: 可以通过检查sun.arch.data.model或os.arch系统属性来获取信息。 64位JVM允许指定的最大堆内存大小为2^64,但实际可用内存会受到操作系统的限制。 JRE、JDK、JVM和JIT之间的区别: JRE代表Java运行时; JDK代表Java开发工具,包含编译器; JVM代表Java虚拟机; JIT代表即时编译。 栈溢出异常发生情况: 如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError; 如果线程递归调用导致无法分配足够的内存,将抛出OutOfMemory异常。

正文

HotSpot 虚拟机对象探秘

对象的创建

Header 解释
使用 new 关键字 调用了构造函数
使用 Class 的 newInstance 方法 调用了构造函数
使用 Constructor 类的newInstance 方法 调用了构造函数
使用 clone 方法 没有调用构造函数
使用反序列化 没有调用构造函数

说到对象的创建,首先让我们看看 Java 中提供的几种对象创建方式:

下面是对象创建的主要流程:


image


虚拟机遇到一条 new 指令时,先检查常量池是否已经加载相应的类,如果没有,必须先执行相

应的类加载。类加载通过后,接下来分配内存。若 Java 堆中内存是绝对规整的,使用“指针碰

撞“方式分配内存;如果不是规整的,就从空闲列表中分配,叫做”空闲列表“方式。划分内存

时还需要考虑一个问题--并发,也有两种方式: CAS 同步处理,或者本地线程分配缓冲(Thread

LocalAllocation Buffer, TLAB)。然后内存空间初始化操作,接着是做一些必要的对象设置(元信

息、哈希码…),最后执行 <init> 方法。


为对象分配内存


类加载完成后,接着会在 Java 堆中划分一块内存分配给对象。内存分配根据Java 堆是否规整,

有两种方式:

  • 指针碰撞:如果 Java 堆的内存是规整,即所有用过的内存放在一边,而空闲的的放在另一

    边。分配内存时将位于中间的指针指示器向空闲的内存移动一段与对象大小相等的距离,这

    样便完成分配内存工作。

  • 空闲列表:如果 Java 堆的内存不是规整的,则需要由虚拟机维护一个列表来记录那些内存

    是可用的,这样在分配的时候可以从列表中查询到足够大的内存分配给对象,并在分配后更

    新列表记录。


选择哪种分配方式是由 Java 堆是否规整来决定的,而 Java 堆是否规整又由所采用的垃圾收集器

是否带有压缩整理功能决定。


image


处理并发安全问题

对象的创建在虚拟机中是一个非常频繁的行为,哪怕只是修改一个指针所指向的位置,在并发情

况下也是不安全的,可能出现正在给对象 A 分配内存,指针还没来得及修改,对象 B 又同时使

用了原来的指针来分配内存的情况。解决这个问题有两种方案:


  • 对分配内存空间的动作进行同步处理(采用 CAS + 失败重试来保障更新操作的原子性);

  • 把内存分配的动作按照线程划分在不同的空间之中进行,即每个线程在Java 堆中预先分配

    一小块内存,称为本地线程分配缓冲(Thread LocalAllocation Buffer, TLAB)。哪个线程

    要分配内存,就在哪个线程的 TLAB 上分配。只有 TLAB 用完并分配新的 TLAB 时,才需要

    同步锁。通过-XX:+/-UserTLAB 参数来设定虚拟机是否使用 TLAB。


image


对象的访问定位

Java 程序需要通过 JVM 栈上的引用访问堆中的具体对象。对象的访问方式取决于 JVM 虚拟机

的实现。目前主流的访问方式有** 句柄** 和 直接指针 两种方式。



  • 指针: 指向对象,代表一个对象在内存中的起始地址。

  • 句柄: 可以理解为指向指针的指针,维护着对象的指针。句柄不直接指向对象,而是指向

    对象的指针(句柄不发生变化,指向固定内存地址),再由对象的指针指向对象的真实内存

    地址。


句柄访问


Java 堆中划分出一块内存来作为句柄池,引用中存储对象的句柄地址,而句柄中包含了对象实例数据对象类型数据各自的具体地址信息,具体构造如下图所示:


image


优势:引用中存储的是稳定的句柄地址,在对象被移动(垃圾收集时移动对象是非常普遍

的行为)时只会改变句柄中实例数据指针,而引用本身不需要修改。


直接指针

如果使用直接指针访问,引用 中存储的直接就是对象地址,那么 Java 堆对象内部的布局中就必

须考虑如何放置访问类型数据的相关信息。


image


优势:速度更,节省了一次指针定位的时间开销。由于对象的访问在 Java 中非常频繁,因此

这类开销积少成多后也是非常可观的执行成本。 HotSpot 中采用的就是这种方式。


64 位 JVM 中,int 的长度是多数?


Java 中,int 类型变量的长度是一个固定值,与平台无关,都是 32 位。意思就是说,在 32 位

和 64 位 的 Java 虚拟机中,int 类型的长度是相同的。


32 位和 64 位的 JVM,int 类型变量的长度是多数?


32 位和 64 位的 JVM 中,int 类型变量的长度是相同的,都是 32 位或者4 个字节。


怎样通过 Java 程序来判断 JVM 是 32 位 还是 64 位?

你可以检查某些系统属性如 sun.arch.data.modelos.arch 来获取该信息。


32 位 JVM 和 64 位 JVM 的最大堆内存分别是多数?


理论上说上 32 位的 JVM 堆内存可以到达 2^32, 即 4GB,但实际上会比这个小很多。不同操

作系统之间不同,如 Windows 系统大约 1.5GB,Solaris大约 3GB。64 位 JVM 允许指定最大

的堆内存,理论上可以达到 2^64,这是一个非常大的数字,实际上你可以指定堆内存大小到

100GB。甚至有的JVM,如 Azul,堆内存到 1000G 都是可能的。


JRE、JDK、JVM 及 JIT 之间有什么不同?


  • JRE 代表 Java 运行时(Java run-time),是运行 Java 引用所必须的。JDK 代表 Java

    发工具(Java development kit),是 Java 程序的开发工具,如 Java 编译器,它也包含

    JRE。

  • JVM 代表 Java 虚拟机(Java virtual machine),它的责任是运行 Java 应用。

  • JIT 代表即时编译(Just In Time compilation),当代码执行的次数超过一定的阈值时,

    会将 Java 字节码转换为本地代码,如,主要的热点代码会被准换为本地代码,这样有利大

    幅度提高Java 应用的性能。


内存溢出异常

Java 会存在内存泄漏吗?


内存泄漏是指不再被使用的对象或者变量一直被占据在内存中。理论上来说,**Java **是有 GC

垃圾回收机制的,也就是说,不再被使用的对象,会被 GC 自动回收掉,自动从内存中清除。


但是,即使这样,Java 也还是存在着内存泄漏的情况,java 导致内存泄露的原因很明确:长

生命周期的对象持有短生命周期对象的引用就很可能发生内存泄露,尽管短生命周期对象已经不

再需要,但是因为长生命周期对象持有它的引用而导致不能被回收,这就是** java** 中内存泄露

的发生场景。


什么情况下会发生栈内存溢出

  1. 栈是线程私有的,他的生命周期与线程相同,每个方法在执行的时候都会创建一个栈帧,用

    存储局部变量表,操作数栈,动态链接,方法出口等信息。局部变量表又包含基本数据类

    型,对象引用类型.

  2. 如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError 异常,

    方法递归调用产生这种结果。

  3. 如果 Java 虚拟机栈可以动态扩展,并且扩展的动作已经尝试过,但是无法申请到足够的内

    存去完成扩展,或者在新建立线程的时候没有足够的内存去创建对应的虚拟机栈,那么

    Java 虚拟机将抛出一个 OutOfMemory 异常。(线程启动过多)

  4. 参数 -Xss 去调整 JVM 栈的大小

与一文了解JVM(中)相似的内容:

一文了解JVM(中)

HotSpot 虚拟机对象探秘 对象的创建 Header 解释 使用 new 关键字 调用了构造函数 使用 Class 的 newInstance 方法 调用了构造函数 使用 Constructor 类的newInstance 方法 调用了构造函数 使用 clone 方法 没有调用构造函数 使用反序

一文了解JVM面试篇(上)

Java内存区域 1、如何解释 Java 堆空间及 GC? 当通过 Java 命令启动 Java 进程的时候,会为它分配内存。内存的一部分用于创建 堆空间,当程序中创建对象的时候,就从对空间中分配内存。GC 是 JVM 内部的一 个进程,回收无效对象的内存用于将来的分配。 2、JVM 的主要组成部分

jvm中类和对象定义存储基础知识

在Java虚拟机中,类和对象是程序的基本组成单元。类定义了一组对象的共性特征和行为,是Java程序中最基本的代码单元。而对象则是具体的实例,有自己独特的状态和行为。在JVM中,类和对象都需要进行存储,因此了解类和对象的存储基础知识对于Java程序员来说是非常重要的。

JVM调优篇:探索Java性能优化的必备种子面试题

本文将带你深入了解JVM调优的重要性、常见问题以及一些实用的调优工具和方法,助你在面试的过程中轻松应对

聊聊JDK19特性之虚拟线程

在读《深入理解JVM虚拟机》这本书前两章的时候整理了JDK从1.0到最新版本发展史,其中记录了JDK这么多年来演进过程中的一些趣闻及引人注目的一些特性,在调研JDK19新增特性的时候了解到了虚拟线程这个概念,于是对虚拟线程进行学习整理内容如下。

从原理聊JVM(一):染色标记和垃圾回收算法

本篇介绍了JVM中垃圾回收器相关的基础知识,后续会深入介绍CMS、G1、ZGC等不同垃圾收集器的运作流程和原理,欢迎关注。

从原理聊JVM(四):JVM中的方法调用原理

多态是Java语言极为重要的一个特性,可以说是Java语言动态性的根本,那么线程执行一个方法时到底在内存中经历了什么,JVM又是如何确定方法执行版本的呢?

带你认识JDK8中超nice的Native Memory Tracking

摘要:从 OpenJDK8 起有了一个很 nice 的虚拟机内部功能: Native Memory Tracking (NMT)。 本文分享自华为云社区《Native Memory Tracking 详解(1):基础介绍》,作者:毕昇小助手。 0.引言 我们经常会好奇,我启动了一个 JVM,他到底会

[转帖]限制容器中的jvm

在rancher中部署完java应用之后,需要对java程序的jvm进行设置,这个非常重要,不然可能会引起比较严重的后果:容器无限制的重启或者主机的内存被耗尽。 在开始之前,先来看一个问题: 在容器中跑了一个java应用,那怎么来限制这个jvm的memory呢? 按照传统的思路对memory进行限制

[转帖]【技术剖析】10. JVM 中不正确的类加载顺序导致应用运行异常问题分析

https://bbs.huaweicloud.com/forum/thread-169439-1-1.html 神Bug... 发表于 2021-11-15 10:36:113973查看 作者:程经纬、谢照昆 > 编者按:两位笔者分享了不同的案例,一个是因为 JDK 小版本升级后导致运行出错,最终