《深入理解Java虚拟机(第3版)-对象创建》

本文深入探讨了Java中创建对象的步骤,包括从常量池获取类信息、内存分配、对象头初始化以及执行构造器。讲解了不同内存分配方式如指针碰撞和空闲列表,以及线程安全问题的解决方案,如CAS和TLAB。此外,还涉及了对象字段的默认初始化和构造器在对象完全构造中的作用。
摘要由CSDN通过智能技术生成

对象的创建,从代码上看,就是new一个对象,或者通过反射调用构造器。

其步骤:

1,获得这个对象所属的类的信息。很好理解,要创建一个对象,那首先得获得这个对象所属的类对吧,从常量池中找到这个类的引用,看类是否已经被加载过。如果没有,则先解析,加载这个类。

        1)为什么从常量池里找?结合JMM的结构,类的信息,版本号,接口,字段,方法等数据,都是放在运行时常量池的。

2,为对象分配内存空间,也就是从堆中划分一部分内存空间用于存储这个对象,这个大小在类加载完了之后就可以确定大小。

        1)为什么这个新对象的大小在类加载完了就可以确认大小?因为类的基本信息,方法,接口,字段,入参,返回参数等,这些都是在加载类的时候已经确认的,是可知的数据类型,其占用的位数,空间也是确定,注意此处还没填充值,只是把数据类型需要占用的空间确定划分下来。

        2))从堆中划分内存空间的方式有两种:

                1)))指针碰撞:假设Java堆中内存是绝对规整的,所有被使用过的内存都被放在一边,空闲的内存被放在另一边,中间放着一个指针作为分界点的指示器,那所分配内存就仅仅是把那个指针向空闲空间方向挪动一段与对象大小相等的距离。

                2)))空闲列表:如果Java堆中的内存并不是规整的,已被使用的内存和空闲的内存相互交错在一起,那就没有办法简单地进行指针碰撞了,虚拟机就必须维护一个列表,记录上哪些内存块是可用的,在分配的时候从列表中找到一块足够大的空间划分给对象实例,并更新列表上的记录。

        3))选取哪一种分配方式?这个由堆是否规整确定?这就很容易让人联想的垃圾回收,像标记清除,就会产生碎片,不规整。复制整理算法则是完整的一片。

        4))线程安全问题:对象创建在虚拟机中是非常频繁的行为,即使仅仅修改一个指针所指向的位置,在并发情况下也并不是线程安全的,可能出现正在给对象A分配内存,指针还没来得及修改,对象B又同时使用了原来的指针来分配内存的情况。

                1)CAS失败重试。

                2)把内存分配的动作按照线程划分在不同的空间之中进行,即每个线程在Java堆中预先分配一小块内存,称为本地线程分配缓冲(ThreadLocalAllocationBuffer,TLAB),哪个线程要分配内存,就在哪个线程的本地缓冲区中分配,只有本地缓冲区用完了,分配新的缓存区时才需要同步锁定。虚拟机是否使用TLAB,可以通过-XX:+/-UseTLAB参数来设定。

3,初始化零值(不包括对象头)。也就是对对象的字段,根据其字段类型设置默认值。这步操作保证了对象的实例字段在Java代码中可以不赋初始值就直接使用,使程序能访问到这些字段的数据类型所对应的零值。

4,设置对象头信息。例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希码(实际上对象的哈希码会延后到真正调用Object::hashCode()方法时才计算)、对象的GC分代年龄等信息。这些信息存放在对象的对象头(ObjectHeader)之中。根据虚拟机当前运行状态的不同,如是否启用偏向锁等。

5,执行构造器方法(执行Class文件中的<init>()方法)。上面的其实可以说都是jvm的底层操作,更多都是操作类与内存,并没有执行程序中的代码。但是这一步就是按照程序员的意愿对对象进行初始化,这样一个真正可用的对象才算完全被构造出来。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值