对象实例化
创建对象方式
-
new
最常见
变形1:XXX的静态方法
变形2:XXXBuilder/XXXFactory的静态方法 -
Class的newInstance:反射方式,只能调用空参的构造器,必须是public
String str = (String) Class.forName("java.lang.String").newInstance();
String str1 = String.class.newInstance();
- Constructor的newInstance(XXX):反射方式,可以调用空参,带参构造器,权限没有要求
Constructor<String> constructor = String.class.getConstructor();
String s1 = constructor.newInstance();
- 使用clone:不需要任何构造器,当前类需要实现CLoneable接口,实现clone
Teacher teacher = new Teacher();
Teacher teacherClone = (Teacher) teacher.clone();
- 反序列化:从文件、网络中获取一个对象的二进制流
//在Teacher类中加入序列号,在序列化和反序列化过程中会检查序列号是否一致
private static final long serialVersionUID = 8024980864121937459L;
// 序列化
ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("object.obj"));
objectOutputStream.writeObject(teacher);
// 反序列化
ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream("object.obj"));
Teacher teacher1 = (Teacher) objectInputStream.readObject();
- 第三方库:Objenesis
创建对象的步骤
-
判断对象对应的类是否加载、链接、初始化
-
为对象分配内存
内存规整-指针碰撞(标记压缩算法)
内存不规整-虚拟机要维护一个列表,空闲列表分配(标记清除算法) -
处理并发安全问题
CAS(Compare and Swap)失败重试、区域加锁保证更新的原子性
给每个线程分配一个TLAB -
初始化分配到的空间:赋默认初始化值,保证对象实例字段不赋值时可以使用
-
设置对象的对象头
-
执行init方法进行初始化(对应类构造器调用)
对象内存布局
对象头(Header)
- 运行时元数据:哈希值、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳
- 类型指针:指向类元数据InstanceClass,确定对象所属类型
实例数据(Instance Data)
- 说明:是对象真正存储的有效信息,包含程序代码中定义的各种类型的字段
- 规则:
1、相同宽度字段被分配在一起;
2、父类中定义的变量会出现在子类之前;
3、如果CompactFields为true,则子类的窄变量可能插入到父类变量的空隙
对齐填充(Padding):占位符作用
package chapter10;
public class Customer {
int id = 1001;
String name;
Account account;
{
name = "匿名用户";
}
public Customer(){
account = new Account();
}
}
class Account{
}
public class TestCustomer {
public static void main(String[] args) {
Customer customer = new Customer();
}
}
对象访问定位
- 句柄访问优点不需要改动栈中地址,缺点占用内存,效率低
- 使用直接指针(HotSpot)优点效率高,不占用内存