本文内容是笔者看了 《深入了解java虚拟机》后的一些笔记,也欢迎各位java爱好者一起学习和交流。
java是一门面向对象的语言,你现在懂怎么面向对象了吗?没对象的抓紧喽,开个小玩笑,开始今天的笔记--对象的创建,在java中,我们听到最多的就是对象了,通过new关键字就可以创建一个对象,那你知道他是如何创建的吗?创建过程中做了哪些事情。
1.虚拟机遇到一条new指令时,首先去检查这个指令的参数能否在常量池中定位倒一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有那就先执行响应的类加载过程。(在后面的笔记中会提到类加载的细节)
------------------------------------------------------检查类是否加载
2.在类加载检查通过后,接下来虚拟机将为新生对象分配内存,对象所需内存的大小在类加载完成后便可以确定。
那么如何分配内存呢?
a.指针碰撞法:假设堆中内存是绝对规整的,所有用过的内存在一边,没用过的内存在另一边,中间放着指针作为分界点的指示器,那分配内存就是将指针向空闲内存方向移动创建对象所需内存大小相等的距离即可。
b.空闲列表:但如果堆中的内存不是规整的,使用的内存和没用使用的内存相互交错,这时虚拟机就必须要维护一个列表,记录哪些内存空间是可用的,哪些内存空间是被占用的。在分配的时从列表中找一块足够大的空间划分给创建的对象实例,并更新列表上的记录。
选择哪种方式区分配内存是由java堆是否规整来决定的,而java堆是否规整又由所采用的垃圾回收器是否带有压缩整理功能来决定的。(还记得上一篇关于堆的介绍吗:java堆可以处于物理上不连续的内存空间,但逻辑上必须连续)
在分配内存时又会遇到一个问题,创建对象是非常频繁的操作,在并发情况下去分配内存是非常不安全的,比如在给A分配内存时,指针还没来得及修改,对象B又使用原来的指针去分配内存,这就将原本要分配给A的内存分配给了B。解决的方案有两种:
a.对分配内存空间的动作做同步处理,实际上虚拟机采用CAS配上失败重试的方式保证更新操作的原子性;
b.另一种是把内存分配的动作按照线程划分在不同的空间中进行,即每个线程在java堆中都预先分配一块内存空间,称为本地线程分配缓存(thread local allocation buffer,tlab),哪个线程要分配内存就在其对应的TLAB上先分配,只有TLAB分配完成时再分配新的TLAB时才需要同步锁定,大大提高了jvm的运行速度。但是默认是不采用这种方式的,需要设置JVM参数-XX;+/UserTLAB来设定开启。
--------------------------------------------------为对象实例分配内存
3.内存分配完成后,虚拟机需要将分配到的内存空间都初始化为零值(不包括对象头),如果使用TLAB,这一工作也提前至TLAB分配时进行。这一操作保证了对象的实例字段在java代码中可以不赋初始化值就可以使用,程序能访问倒这些字段的数据类型所对应的零值。记住哦,你所使用的初始值都是初始化时分配的
--------------------------------------------------为对象进行零值初始化
4.接下来,虚拟机要对对象做必要的设置:例如这个对象是哪个类的实例、如何才能找到类的元数据、对象的哈希码、对象的GC分代年龄等信息,这些信息都存放在对象头(Object header)中。
--------------------------------------------------初始化对象头
5.在上面的工作完成之后,从虚拟机的视角来看,一个新的对象已经产生了。但从java程旭员的角度来看,对象创建还少了自定义的一步,也就是我们对对象的一个初始化。接下来就是执行<init>方法,按照程序员的意志来初始化对象。
--------------------------------------------------初始化对象
6.将对象引用入栈。
--------------------------------------------------将对象引用入栈
当它本可进取时,却故作谦卑;
当它在空虚时,用爱欲来填充;
在困难和容易之间,它选择了容易;
它犯了错,却借由别人也会犯错来宽慰自己;
它自由软弱,却把它认为是生命的坚韧;
当它鄙夷一张丑恶的嘴脸时,却不知那正是自己面具中的一副;
它侧身于生活的污泥中,虽不甘心,却又畏首畏尾。