都知道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. VS2010-MFC(常用控件:图片控件Picture Control)

    转自:http://www.jizhuomi.com/software/193.html 本节主要讲一种简单实用的控件,图片控件Picture Control.我们可以在界面某个位置放入图片控件,显示 ...

  2. java.sql.SQLSyntaxErrorException: ORA-00932: 数据类型不一致: 应为 NUMBER, 但却获得 BINARY

    2019-07-24 17:24:35 下午 [Thread: http-8080-4][ Class:net.sf.ehcache.store.disk.Segment Method: net.sf ...

  3. hexo next主题中遇到的关于require的timeout的问题。Uncaught Error: Load timeout for modules:

    个人博客:https://mmmmmm.me 源码:https://github.com/dataiyangu/dataiyangu.github.io 背景 报错:Uncaught Error: L ...

  4. Mysql优化系列之查询性能优化前篇1

    前言 这是优化系列的最后一篇的第1小篇,我们其实可以直接从sql怎么写讲起,why not?但是我还是决定花2个篇幅 问一些问题,带着几个问题循序渐进的往下走. 一个sql语句是怎么被执行的? sql ...

  5. 17.splash_case02

    # 抓取<我不是药神>的豆瓣评论 import csv import time import requests from lxml import etree fw = open('doub ...

  6. 次短路 /// dijkstra oj1597

    题目大意: 给出一个有向图,求从 顶点a 到 顶点b 的次短路. 第一行是2个正整数 n 和 e,表示该有向图的顶点数和边数.3 < n ≤ 5000 , 3 < e < 40000 ...

  7. visual studio 注释模板

    /*----------------------------------------------------------------* 项目名称 :$rootnamespace$* 项目描述 :* 类 ...

  8. java笔试之自守数

    自守数是指一个数的平方的尾数等于该数自身的自然数.例如:25^2 = 625,76^2 = 5776,9376^2 = 87909376.请求出n以内的自守数的个数 接口说明 /*功能: 求出n以内的 ...

  9. Java开发系列-文件上传

    概述 Java开发中文件上传的方式有很多,常见的有servlet3.0.common-fileUpload.框架.不管哪种方式,对于文件上传的本质是不变的. 文件上传的准备 文件上传需要客户端跟服务都 ...

  10. css3 随记

    1 让子元素对其的方式  box-pack 2 -webkit-text-size-adjust  解决字体大小失效问题http://www.frontopen.com/273.html 3 disp ...