都知道Java是一门面向对象的编程语言,在Java程序运行过程中,无时无刻不在创建对象,所以这节来总结一下HotSpot虚拟机中的Java对象。

一、Java虚拟机对象的创建过程。

在语义层面创建一个对象就是new的过程,但是在虚拟机里面,对象(这里的对象指的是普通对象,不包括数组和class对象等)又是怎么被创建的呢?

1、当Java虚拟机收到一个new指令时,它首先会去常量池中检查是否存在这个类的符号引用。如果存在则会执行下一步,否则会创建一个符号引用放在常量池中(待考察);

2、当在常量池中找到类的符号引用之后,还需要判断这个类是否已经被加载、初始化过,如果有则执行下一步,否则进行类加载(以后介绍详细加载过程);

3、如果类已经加载,那么这时候就已经知道它所需要的空间大小,于是就可以为之分配内存;

4、分配内存完成后,虚拟机需要将分配到的内存初始化为0值(不包括对象头),这一操作保证了对象的实例字段在Java代码中不需要赋初值就能直接使用,程序能访问到这些字段的数据类型所对应的零值。

5、接下来,虚拟机要对对象进行必要的设置。例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象哈希码、对象的GC分代年龄等信息 。这些信息一般被称为对象的头信息,用来关联对象与它所属的类型。

6、在以上步骤都完成之后,从虚拟机的视角来看,一个新的对象已经被创建,但是从Java程序的视角来看,对象的创建才刚刚开始,因为初始化操作init方法还没有被执行,所有的字段都还为零,所以new指令之后,会执行init方法,把对象按照程序员的意愿进行初始化,这样一个可用的对象才算完全产生出来。

从以上过程来看,一个对象的创建可以从两个视角来分析,一个是Java虚拟机视角,另一个是程序员视角,明显程序员视角是比较简单的,因为我们不需要关心类加载、内存分配及对象头设置等这些【底层】的过程,其实也不是说不需要关心,虽然我们不去操作这些过程,但还是有必要知道这些原理。

二、对象的内存布局

一个对象被创建之后,它在内存中是怎样存储的呢?在HotSpot虚拟机中,内存的布局分为三部分:

1、对象头。

对象头也包括两部分,一部分是用于存储对象自身的运行数据,如哈希码、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳等;另一部分是类型指针,即对象指向它的类元数据的指针,虚拟机通过这个指针来确定对象时哪个类的实例;

2、实例数据部分。

实例数据部分才是对象真正存储的有效信息,也是程序代码中所定义的各种类型的字段内容,包括从父类继承下来的和在子类中定义的。

3、对齐填充。并不是必然存在,也没有特别的意义,仅仅起着占位符的作用。由于在HotSpot VM中的自动内存管理系统要求对象的起始地址必须是8字节的整数倍,也就是说对象的大小必须是8字节的整数倍,而对象头部分正好是8字节整数倍(1倍或2倍),因此,当对象实例数据部分没有对齐时,就需要通过对齐填充来补全。

三、对象的访问定位

建立对象时为了访问对象,Java程序通过栈上的reference数据来操作堆上的实例对象。由于reference类型在Java规范中只规定了一个指向对象的引用,并没有定这个引用要以何种方式去定位、访问堆中对象的具体位置,所以对象方法的方式也是取决于虚拟机实现的方式。目前主流的访问方式有使用句柄和直接指针两种。

1)如果使用句柄访问的话,Java堆中将开辟一块内存用来作为句柄池,存放句柄,reference存储的就是句柄池的地址,而句柄中则包含了对象实例数据(堆内存)和对象类型数据,如下图所示:

2)如果使用直接指针访问,那么reference直接指向堆内存中的对象实例数据,而在对象实例数据中开辟一块内存存放指向对象类型数据地址的句柄,如下图所示:

这两种对象访问方式各有优势,

使用句柄访问最大的好处就是reference中存储的是稳定的句柄地址,在对象被移动时只改变句柄中的实例数据指针即可,reference本身不需要改变;

使用直接访问方式最大的好处就是速度更快,它节省了一次对象实例数据指针定位的时间开销,因为reference指向的就是实例数据,但是如果对象被移动的话,就得改变reference才能保证对象正确的引用。

以上就是HotSpot虚拟机中对象的创建过程、布局及访问方式的总结,因没有安装visio,图是用word画的,想死的心都有了。。。。

备注:以上内容来源于《深入理解Java虚拟机》,根据自己的理解来总结,有些内容和书本内容相同是因为只有这么描述才比较好懂,毕竟大神还是大神,是颜色不一样的烟火!!

Java虚拟机系列(二)---HotSpot虚拟机对象的更多相关文章

  1. java‘小秘密’系列(二)---Integer

    java'小秘密'系列(二)---Integer 前言:本系列的主题是平时容易疏忽的知识点,只有基础扎实,在编码的时候才能更注重规范和性能,在出现bug的时候,才能处理更加从容. 目录 java'小秘 ...

  2. Java 虚拟机系列二:垃圾收集机制详解,动图帮你理解

    前言 上篇文章已经给大家介绍了 JVM 的架构和运行时数据区 (内存区域),本篇文章将给大家介绍 JVM 的重点内容--垃圾收集.众所周知,相比 C / C++ 等语言,Java 可以省去手动管理内存 ...

  3. Java 虚拟机 - 2.3 HotSpot虚拟机对象

    对象的创建 Step1 类加载检查 当发现一条new指令时,检查: 该指令的参数是否能在常量池中定位到一个类的符号引用: 并且检查这个符号引用代表的类是否已经被加载.解析和初始化过.如果没有,那必须先 ...

  4. java多线程系列(二)---对象变量并发访问

    对象变量的并发访问 前言:本系列将从零开始讲解java多线程相关的技术,内容参考于<java多线程核心技术>与<java并发编程实战>等相关资料,希望站在巨人的肩膀上,再通过我 ...

  5. java多线程系列(二)

    对象变量的并发访问 前言:本系列将从零开始讲解java多线程相关的技术,内容参考于<java多线程核心技术>与<java并发编程实战>等相关资料,希望站在巨人的肩膀上,再通过我 ...

  6. Java爬虫系列二:使用HttpClient抓取页面HTML

    爬虫要想爬取需要的信息,首先第一步就要抓取到页面html内容,然后对html进行分析,获取想要的内容.上一篇随笔<Java爬虫系列一:写在开始前>中提到了HttpClient可以抓取页面内 ...

  7. 乐字节Java反射之二:实例化对象、接口与父类、修饰符和属性

    大家好,小乐继续接着上集:乐字节Java反射之一:反射概念与获取反射源头Class 这次是之二:实例化对象.接口与父类.修饰符和属性 一:实例化对象 之前我们讲解过创建对象的方式,有new .克隆.反 ...

  8. 深入理解java虚拟机系列二——垃圾收集算法

    在主流的商用程序语言中大多都是用根搜索算法(GC Roots Tracing)判断对象是否存活,比如java,c#等.当从GC Roots到某个对象不可达,则证明此对象是不可用的,将要被回收. 商业虚 ...

  9. java教程系列二:Java JDK,JRE和JVM分别是什么?

    多情只有春庭月,犹为离人照落花. 概述 本章主要了解JDK,JRE和JVM之间的区别.JVM是如何工作的?什么是类加载器,解释器和JIT编译器.还有一些面试问题. Java程序执行过程 在深入了解Ja ...

  10. Java基础系列二:Java泛型

    该系列博文会告诉你如何从入门到进阶,一步步地学习Java基础知识,并上手进行实战,接着了解每个Java知识点背后的实现原理,更完整地了解整个Java技术体系,形成自己的知识框架. 一.泛型概述 1.定 ...

随机推荐

  1. "一个实用的却被忽略的命名空间:Microsoft.VisualBasic":

        当你看到这个命名空间的时候,别因为是vb的东西就匆忙关掉网页,那将会是您的损失,此命名空间中的资源最初目的是为了简化vb.net开发而创建的,所以microsoft.visualbasic并不 ...

  2. VS2010-MFC(常用控件:按钮控件的编程实例)

    转自:http://www.jizhuomi.com/software/184.html 因为Button控件在前面的例子中涉及到了,比较简单,本文就不作深入分析了,而是重点讲解单选按钮Radio B ...

  3. Linux复制指令

    功能: 复制文件或目录说明: cp指令用于复制文件或目录,如同时指定两个以上的文件或目录,且最后的目的地是一个已经存在的目录,则它会把前面指定的所有文件或目录复制到此目录中.若同时指定多个文件或目录, ...

  4. 《DSP using MATLAB》Problem 8.34

    今天下了小雨,空气中泛起潮湿的味道,阴冷的感觉袭来,心情受到小小影响. 代码: hp2lpfre子函数 function [wpLP, wsLP, alpha] = hp2lpfre(wphp, ws ...

  5. [PKUSC2018]神仙的游戏

    题目 画一画就会发现一些奇诡的性质 首先如果\(len\)为一个\(\operatorname{border}\),那么必然对于\(\forall i\in [1,len]\),都会有\(s_i=s_ ...

  6. 13-5-let和()的作用域

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  7. java_IO流(输出流)

    ** * io流: * 输入流:硬盘输入到内存 字节/字符输入流 * 输出流:内存输出到硬盘 字节/字符输入流 * 字节流:一切数据都是字节存储(二进制) * 字节输出流(OutputStream): ...

  8. spring:bean的作用范围和生命周期

    bean的作用范围调整: <!--bean的作用范围调整 bean标签的scope属性: 作用:用于指定bean的作用范围 取值:常用的就是单例的和多例的 singleton:单例的(默认值) ...

  9. [复习]平衡树splay

    明天要考试了…… 出来写一个splay的复习总结. 怕忘…… ^废话^ 以下内容学习自yyb大神的博客, 由于yyb大神内容不全, 部分是博主本人自行脑补... 这个模板还是比较全的^-^ ^又是一堆 ...

  10. 企业微信开发免登授权时提示scope不能为空,错误代码1001

    企业免登授权提示scope不能为空1001 原因是我们是单页面应用url自带#/在微信里面认为#号后面的参数不被识别 后端开发人员把参数放到跳转 URL地址前面,正确形式是 https://open. ...