java对象的创建

java对象的创建

image

在语言层面上,创建对象(例如克隆、发序列化)通常仅仅是一个new关键字而已。

类加载过程

而虚拟机遇到一个new指令时,首先去检查这个指令的参数是否能在常量池中定义一个定位到一个类的引用,并检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有,那必须先执行相应的类加载过程。

在类加载检查通过后,接下来虚拟机要为新生对象分配内存。对象所需的内存大小在类加载完成后便可以完全确定,为对象分配内存空间的任务相当于把一块确定大小的内存从java堆中划分出来。

为对象分配内存

分配方式
  1. 假设java堆是绝对规整的,所有用过的内存都放在一遍,所有空闲的内存放在另一边,中间放一个指针作为分界点的指示器,那所分配的内存就是仅仅把指针向空闲的内存移动与新生对象大小相等的距离,这种分配方式叫做指针碰撞
  2. 假设java堆不是绝对规整的,就不能用指针碰撞了,虚拟机就必须维护一个列表,记录上哪块内存是可用的,在分配内存是找到一个足够大的空间分配给对象实例,并更新列表上的记录,这种方式成为空闲列表

选择哪种方式有java堆是否规整决定,而java堆是否规整又由所采用的垃圾回收器是否带有压缩整理功能决定。所以,使用Serial、ParNew等带Compact过程的收集器时,JVM采用指针碰撞方式分配内存;而使用CMS这种基于标记-清除(Mark-Sweep)算法的收集器时,采用空闲列表方式。

线程安全问题

除如何划分可用空间之外,还有另一个需要考虑的问题是对象创建在虚拟机中是否非常频繁的行为,即使仅仅修改一个指针所指向的位置,在并发情况下也并不是线程安全的,可能出现正在给A分配内存,指针还没来得及修改,对象B又用了原来的指针进行分配内存的情况。解决这个问题有两种解决方案:

同步处理

JVM采用CAS(Compare and
Swap)机制加上失败重试的方式,保证更新操作的原子性。
CAS:有3个操作数,内存值V,旧的预期值A,要修改的新值B。当且仅当预期值A和内存值V相同时,将内存值V修改为B,否则什么都不做。

本地线程分配缓冲区

在每个线程在Java堆预先分配一小块内存,称为本地线程分配缓冲区(Thread Local Allocation Buffer,TLAB)。
哪个线程需要分配内存就从哪个线程的TLAB上分配。
只有TLAB用完需要分配新的TLAB时,才需要同步处理。
JVM通过”-XX:+/-UseTLAB”指定是否使用TLAB。

对象内存初始化为零

对象内存初始化为零,但不包括对象头。
如果使用TLAB,提前至分配TLAB时。
这保证了程序中对象(及实例变量)不显式初始赋零值,程序也能访问到零值。

设置对象头

虚拟机要对对象进行必要的设置,例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希吗、对象的GC分代年龄等信息。这些信息存放在对象的对象头之中。根据虚拟机当前的运行状态的不同,如是否启用偏向锁等,对象头会有不同的设置方式。

执行对象实例方法

在上面工作都完成之后,从虚拟机的视角来看,一个新的对象已经产生了,但从Java程序的视角来看,对象的创建才刚刚开始–方法还没有执行,所有的字段都还为零。所以,一般来说(有字节码中是否跟随invokespecial指令所决定的),执行new指令之后会接着执行方法,把对象按照程序员的意愿进行初始化,这样的一个真正可用的对象才算完全产生出来。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值