都知道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. C++函数调用原理理解

    空程序: int main() { 00411360  push        ebp       ;压入ebp 00411361  mov         ebp,esp     ;ebp = es ...

  2. ps快速将白底图片变为透明图片

    方法一: 如果图层有锁图标,则要点击它,然它消失.然后选中魔棒工具,然后点击图片上要透明的区域,按下backspace键即可. 方法二: 转载自:https://blog.csdn.net/sunyi ...

  3. Resource Archiver HDU - 3247 AC自动机+BFS+状压

    题意: 给出n个资源串,m个病毒串,现在要如何连接资源串使得不含病毒串(可以重叠,样例就是重叠的). 题解: 这题的套路和之前的很不同了,之前的AC自动机+DP的题目一般都是通过teir图去转移, 这 ...

  4. ssm项目中使用拦截器加上不生效解决方案

    在很多时候,需要拦截器来帮助我们完成一些特定的工作,比如获取请求的参数,本身在request这种获取数据就是一次磁盘的io, 如果在filter中获取了参数,那么在controller中就不能获取相关 ...

  5. docker上安装ActiveMQ

    1.查看是否已经存在镜像 docker images 2.搜索镜像 docker search activemq 3.拉取镜像 docker pull webcenter/activemq 4.构建容 ...

  6. Hadoop2.7.1配置NameNode+ResourceManager高可用原理分析

    关于NameNode高可靠需要配置的文件有core-site.xml和hdfs-site.xml 关于ResourceManager高可靠需要配置的文件有yarn-site.xml 逻辑结构: Nam ...

  7. (转)NodeJS收发GET和POST请求

    NodeJS收发GET和POST请求 目录: 一 express框架接收 二 接收Get 三 发送Get 四 接收Post 五 发送Post 一 express框架接收 1 2 3 4 5 app.g ...

  8. opencv-图像遍历

    #include "stdafx.h" #include<opencv2/opencv.hpp> #include<iostream> #include&l ...

  9. openssl操作公私钥和加解密的一些常用命令

    生成公私钥实践: 生成私钥,这里以椭圆曲线secp256k1为例: openssl ecparam -name secp256k1 -genkey -out secp256k1-priv.pem #带 ...

  10. 三. var let const的理解 以及 立即执行函数中的使用 以及 for循环中的例子

    一. 立即执行函数 windows中有个name属性,name='' '' var 如果我们用var name 去声明,那就会改变windows中name的值(因为我们不是在函数作用域中声明的,所以会 ...