对象实例和对象引用
关键字new用来创建对象。从一个类中创建一个对象被称为实例化,所以,对象也常被称为实例。Java中新创建的对象被放在被称为“堆”的系统存储区中。Java中的对象通过“对象引用”(object reference)来访问。一个类的变量持有一个对象,实际就是对该对象的引用。当变量不再引用任何对象时,该对象引用就为null。
Object obj=new Object();
//Object:类
//obj:对象的引用
//new Object():新建对象
public static void main(String[] args) {
String str;//一个对象引用str指向零个对象
str=new String("string1");//一个对象引用str指向一个对象string1
str=new String("string2");//注意:这里对象引用str并不是指向第二个对象string2,而是将之前指向第一个对象string1的引用重新指向了另一个对象string2
}
一个对象的引用同一时间只能指向零个或1个对象,而一个对象可以同时被多个引用变量引用。引用类型变量就相当于电脑上的快捷方式;对象就相当于你磁盘里面安装的游戏,它实实在在占用你的空间内存; 而变量只是快捷方式。
对象的内存布局
HotSpot虚拟机中,对象在内存中存储的布局可以分为三块区域:对象头(Header)、实例数据(Instance Data)和对齐填充(Padding)。
对象头
Mark Word:比如 hash码,对象所属的年代,对象锁,锁状态标志,偏向锁(线程)ID,偏向时间。
Kclass pointer:对象实例在堆中,对象所属类信息在方法区,那如何访问方法区的数据信息呢?就是通过这个元数据指针。在对一个类进行初次加载时会生成一个类对象,然后通过反射的方式获取类的信息,反射是通过元数据指正拿到信息的。(这个类对象和new一个该类的对象,不是同一个,如果只加载没有new,那这个对象未进行初始化)。
数组长度(只有数组对象才有):如果对象是数组类型,因为JVM虚拟机可以通过Java对象的元数据信息确定Java对象的大小,但是无法从数组的元数据来确认数组的大小,所以用一块来记录数组长度。
实例数据:存放类的属性数据信息,包括父类的属性信息;
对齐填充:由于虚拟机要求 对象起始地址必须是8字节的整数倍。填充数据不是必须存在的,仅仅是为了字节对齐。
参考博客:(4条消息) 对象内存布局详解_@来杯咖啡的博客-CSDN博客_对象的内存布局
Java new一个对象过程
确认类元信息是否存在,当JVM执行引擎接收到new指令时,首先在元空间内检查需要创建的类元信息是否存在。若不存在,那么在双亲委派模式下,使用当前类加载器以 ClassLoader + 包名+类名为 Key 进行查找对应的 class文件。 如果没有找到文件,则抛出 ClassNotFoundException 异常,如果找到,则进行类加载(加载 - 验证 - 准备 - 解析 - 初始化),并生成对应的 Class 类对象。类加载阶段的初始化是初始化类变量(静态变量)。
分配对象内存,首先计算对象占用空间大小,如果实例成员变量是引用变量,仅分配引用变量空间即可,即4个字节大小,接着在堆中划分一块内存给新对象。 在分配内存空间时,需要进行同步操作,比如采用 CAS (Compare And Swap)失败重试、 区域加锁等方式保证分配操作的原子性。
设定默认值,成员变量值都需要设定为默认值, 即各种不同形式的零值。(这里的成员变量存于堆中作为对象的实例数据,静态变量在类加载时就存于方法区,局部变量则是等执行方法时加载到局部变量表,事先一个还是存于class文件,变量的操作都是对其引用进行操作,真实数据值在常量池中)
设置对象头,设置新对象的哈希码、 GC 信息、锁信息、对象所属的类元信息等。这个过程的具体设置方式取决于 JVM 实现。
执行init方法,初始化该对象成员变量,执行实例化代码块,调用类的构造方法,并把堆内对象的首地址赋值给引用变量。