JVM虚拟机笔记(二)--HotSpot对象的创建

对象的创建过程:

当虚拟机遇到一条含有new的指令时,会进行一系列对象创建的操作:

1、检查常量池中是否有即将要创建的这个对象所属类的引用;

  (1)若常量池中有没有这个类的符号引用,说明这个类还没有被定义!抛出ClassNotFoundException;

  (2)若常量池中有这个类的符号引用,则进行下一步操作:

2、进而检查这个符号引用所代表的类是否已经被JVM加载:

  (1)若该类没有被加载,就找该类的class文件,并加载进方法区;

  (2)如该类已经被JVM加载,则准备为对象分配内存;

3、根据方法区中该类的信息确定该泪所需的内存大小;

  一个对象所需的内存大小是这个对象所属类被定义完就能确定的!且一个类所产生的所有对象的内存大小是一样的!

  JVM在一个类被加载进方法区的时候就知道该类生产的每一个对象所需要的内存大小。

  1. 从堆中划分一块对应大小的内存空间给新的对象;

    分配堆中内存有两种方式:

    • 指针碰撞

      如果JVM的垃圾收集器采用复制算法或标记-整理算法,那么堆中空闲内存是完整的区域,并且空闲内存和已使用内存之间由一个指针标记。那么当为一个对象分配内存时,只需移动指针即可。因此,这种在完整空闲区域上通过移动指针来分配内存的方式就叫做“指针碰撞”。

    • 空闲列表

      如果JVM的垃圾收集器采用标记-清除算法,那么堆中空闲区域和已使用区域交错,因此需要用一张“空闲列表”来记录堆中哪些区域是空闲区域,从而在创建对象的时候根据这张“空闲列表”找到空闲区域,并分配内存。

      综上所述:JVM究竟采用哪种内存分配方法,取决于它使用了何种垃圾收集器。

  2. 为对象中的成员变量赋上初始值(默认初始化);
  3. 设置对象头中的信息;
  4. 调用对象的构造函数进行初始化

    此时,整个对象的创建过程就完成了。

对象的内存模型

一个对象从逻辑角度看,它由成员变量和成员函数构成,从物理角度来看,对象是存储在堆中的一串二进制数,这串二进制数的组织结构如下。

对象在内存中分为三个部分:

  1. 对象头
  2. 实例数据
  3. 对齐补充

1. 对象头

对象头中记录了对象在运行过程中所需要使用的一些数据:哈希码、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳等。

此外,对象头中可能还包含类型指针。通过该指针能确定这个对象所属哪个类。

此外,如果对象是一个数组,那么对象头中还要包含数组长度。

2. 实例数据

实力数据部分就是成员变量的值,其中包含父类的成员变量和本类的成员变量。

3. 对齐补充

用于确保对象的总长度为8字节的整数倍。

HotSpot要求对象的总长度必须是8字节的整数倍。由于对象头一定是8字节的整数倍,但实例数据部分的长度是任意的,因此需要对齐补充字段确保整个对象的总长度为8的整数倍。

访问对象的过程

我们知道,引用类型的变量中存放的是一个地址,那么根据地址类型的不同,对象有不同的访问方式:

  1. 句柄访问方式

    堆中需要有一块叫做“句柄池”的内存空间,用于存放所有对象的地址和所有对象所属类的类信息。

    引用类型的变量存放的是该对象在句柄池中的地址。访问对象时,首先需要通过引用类型的变量找到该对象的句柄,然后根据句柄中对象的地址再访问对象。

  2. 直接指针访问方式

    引用类型的变量直接存放对象的地址,从而不需要句柄池,通过引用能够直接访问对象。

    但对象所在的内存空间中需要额外的策略存储对象所属的类信息的地址。

比较

HotSpot采用直接指针方式访问对象,因为它只需一次寻址操作,从而性能比句柄访问方式快一倍。但它需要额外的策略存储对象在方法区中类信息的地址。

原文地址:https://www.cnblogs.com/yplq/p/8985119.html

时间: 2024-08-28 20:44:05

JVM虚拟机笔记(二)--HotSpot对象的创建的相关文章

Java虚拟机(二)对象的创建与OOP-Klass模型

相关文章 Java虚拟机系列 前言 在前一篇文章中我们学习了Java虚拟机的结构原理与运行时数据区域,那么我们大概知道了Java虚拟机的内存的概况,那么内存中的数据是如何创建和访问的呢?这篇文章会给你答案. 1.对象的创建 对象的创建通常是通过new一个对象而已,当虚拟机接收到一个new指令时,它会做如下的操作. (1)判断对象对应的类是否加载.链接.初始化 虚拟机接收到一条new指令时,首先会去检查这个指定的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被类加载

深入java虚拟机(二) 对象的创建

java创建对象通常的方式是使用new指令,虚拟机会首先检查new指令的参数(也就是new关键字后面跟着的类名)是否能够在常量池中找到一个类的符号引用,并根据这个符号引用检查其代表的类是否已经加载.解析和初始化,如果没有就先执行类的加载过程.类加载检查后,就会给对象分配内存.新生的对象一般会存在于java堆中,根据java堆的情况,分配内存主要分为两种方式:"指针碰撞"(bump the pointer)和"空闲列表"(free list). 指针碰撞:假设java

深入理解JVM读书笔记二: 垃圾收集器与内存分配策略

3.2对象已死吗? 3.2.1 引用计数法 给对象添加一个引用计数器,每当有一个地方引用它的地方,计数器值+1:当引用失效,计数器值就减1;任何时候计数器为0,对象就不可能再被引用了. 它很难解决对象之间相互循环引用的问题. 3.2.2 可达性分析算法 这个算法的基本思路就是通过一系列的称为"GC Roots"的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链(Reference Chain),当一个对象到GC Roots没有任何引用链相连(用图论的话来说,就是从GC

jvm学习笔记二(减少GC开销的建议)

一:触发主GC(Garbage Collector)的条件 JVM进行次GC的频率很高,但因为这种GC占用时间极短,所以对系统产生的影响不大.更值得关注的是主GC的触发条件,因为它对系统影响很明显.总的来说,有两个条件会触发主GC: 1)当应用程序空闲时,即没有应用线程在运行时,GC会被调用.因为GC在优先级最低的线程中进行,所以当应用忙时,GC线程就不会被调用,但以下条件除外. 2)Java堆内存不足时,GC会被调用.当应用线程在运行,并在运行过程中创建新对象,若这时内存空间不足,JVM就会强

垃圾回收算法简介——JVM读书笔记<二>

垃圾回收的过程主要包括两部分:找出已死去的对象.移除已死去的对象. 确定哪些对象存活有两种方式:引用计数算法.可达性分析算法. 方案一:引用计数算法 给对象中添加一个引用计数器,每当有一个地方引用它时,计数器值加1:当引用失效时,计数器值减1:计数器的值为0时即表明对象已经死去(可被回收). 优点:实现简单,判定效率高. 缺点:难解决对象之间互相引用的问题. 如:对象objA和objB都有字段instance,令objA.instance=objB,objB.instance=objA;除此之外

java学习笔记——java中对象的创建,初始化,引用的解析

如果有一个A类. 1.例如以下表达式: A  a1 = new A(); 那么A是类,a1是引用.new A()是对象.仅仅是a1这个引用指向了new A()这个对象. 2.又如: A  a2; A代表类,a2仅仅是一个引用,仅仅是a2指向的对象为空NULL. 3.再如: a2 = a1: 它代表的是a2是一个引用,a1也是一个引用:在这个过程中a1所指向对象的地址传给了a2,使得a2.a1指向同一对象. 4.引用 如以下代码片段: int  k; // base data type A  a;

java之jvm学习笔记二(类装载器的体系结构)

java的class只在需要的时候才内转载入内存,并由java虚拟机的执行引擎来执行,而执行引擎从总的来说主要的执行方式分为四种, 第一种,一次性解释代码,也就是当字节码转载到内存后,每次需要都会重新的解析一次, 第二种,即时解析,也就是转载到内存的字节码会被解析成本地机器码,并缓存起来以提高重用性,但是比较耗内存, 第三种,自适应优化解析,即将java将使用最贫乏的代码编译成本地机器码,而使用不贫乏的则保持字节码不变,一个自适应的优化器可以使得java虚拟机在80%-90%的时间里执行优化过的

深入理解Java虚拟机笔记---引用类型和对象是否死亡

在JDK1.2以前,Java中的引用定义得很传统:如果reference类型的数值代表的是另外一块内存的起始地址,就称这块内存代表中一个引用.这种定义很纯粹,但太过狭隘,一个对象在这种定义下只有被引用或者没有引用两种状态,对于如何描述一个"食之无味,弃之可惜"的对象就显得无能为力:如果内存在进行垃圾收集后还是非常紧张,则可以抛弃这些对象.很多系统的缓存功能都符合这样的应用场景. 在JDK1.2之后,Java对引用的概念进行了扩充,将引用分为强引用(Strong Reference),软

《COM原理与应用》学习笔记二——COM对象和COM接口的实现

COM对象是给用户提供服务的封装的实体.这个应该和C++中类的对象理解起来是相似的.但是有时候也把COM对象当作提供服务的那个类.COM对象也对数据进行了封装,然后也提供了接口.不过和类还是有一些不一样的.类中的数据可以申明为public,然后让用户能够直接访问这些数据成员.但是用户不能对COM对象的数据进行直接访问,只能通过接口(如果有提供这种接口的话)来对数据进行间接的访问.一般COM接口指的是一组提供服务的接口,刚开始看这个定义很不习惯.因为C++中根本没有接口的概念,但是像Java这些语