对象的实例化
这部分大厂面试的时候也经常问这些东西。比如
对象的创建方式
- new
最常见的方式
变形构造器被私有化, 有一个返回对象的静态方法
变形2:xxxBuilder/xxxFactor - class的newInstance()方法:反射的方式, 只能调用空参构造器, 权限必须时public
- Constructor的newInstance(xx):反射的方式, 可以调用空参、带参的构造器, 权限没有要求
- 使用clone(): 不用空参构造器, 当前类要实现Cloneable接口, 实现方法
- 使用反序列化::从文件或者从网络中获取一个对象的二进制流。
- 第三方库Objenesis
对象创建的步骤
- 字节码层面
- 先执行new 判断当前类是否加载, 没有加载, 利用类的加载器加载, 在堆中开辟空间, 默认初始化值
- dup复制, 压入操作数栈
- invokespecial调用 init方法就是构造器显式赋值
- astore_1存到局部变量表
- 从执行步骤的角度来分析
-
判断对象对应的类是否加载、链接、初始化(加载类元信息)
-
为对象分配内存
- 如果堆空间内存规整
- 利用指针碰撞: 非空闲的内存放在一边, 空闲的内存在另一边, 中间用指针分开, 分配内存仅仅是把指针移动跟对象大小相等的距离, 一般垃圾收集器选择Serial、ParNew这个基于压缩算法的(内存规整), 就就采用这种方法。
- 内存不规整
内存不规整, 建立空闲分区列表, 从可用的空闲分区, 中找到大于当前对象大小的区域放进去区。(CMS垃圾回收器) - 使用那种分配算法看使用的垃圾回收器是否又压缩整理功能决定的
- 如果堆空间内存规整
-
- 处理并发安全问题:
- 采用CAS失败重试、区域加锁保证更新的原子性
- 每个线程预先分配一块TLAB在jdk8以及默认开启
- 初始化分配空间
- 给对象属性默认初始化
- 设置对象的对象头
对象的所属类(即类的元数据信息)、对象的hashCode和对象的GC信息, 锁信息等数据存储在对象的对象投中, 这个过程的具体设置方式取决于jvm实现 - 执行init方法进行初始化
- 显式初始化,执行构造器
对象的内存布局
对象头
包含两部分
- 运行时元数据 (用Markword, 在32位虚拟机时32 在64位时64位)
- hash值: 初始地址值
- GC分代年龄: age
- 锁状态标志
- 线程持有的锁
- 偏向线程ID
- 偏向时间戳
- 类型指针 (32位虚拟机32 64位虚拟机 64)
指向类元数据的InstanceKlass, 确定该对象所属的类型 - 如果是数组还需记录数组长度
实例数据
他是真正存储的有效信息, 包括程序代码中定义的各种类型的字段(包括从父类继承下来和本身拥有的字段)
相同宽度的字段总是被分配到一起
父类定义的变量会出现在子类之前
如果ComPactFields参数为true(默认为true):子类的窄变量可能插入到父类变量的空析
对齐填充
因为JVM要求java的对象占的内存大小应该是8bit的倍数,所以后面有几个字节用于把对象的大小补齐至8bit的倍数,没有特别的功能。
对象访问定位
- 创建对象的目的是为了使用它
- jvm是如何通过栈帧中的对象引用访问到其内部的对象实例的呢: 通过栈中的引用
- 对象访问方式主要有两种
-
句柄访问(两次访存)
- 便于移动而对于栈的引用是不需要修改
- 便于移动而对于栈的引用是不需要修改
-
直接指针(hotspot使用)(效率高)
-