(1) 虚拟机接收到一条new指令时,先去虚拟机中检查这个指令的参数是否能在常量池中定位到一个类的符号引用,即类有没有被加载到方法区;
(2) 若类未被加载到方法区,则先进行类加载,若类已被加载,则继续;
(3) 获取被加载的类的对象长度;
(4) 确认是否在TLAB中分配内存,若是,则在TLAB中分配内存,否则在EDEN中分配内存;
(5) 将分配到的内存空间设置为零值;
(6) 设置对象的头信息;
(7) 将对象的引用入虚拟机栈;

 

内存分配方式:
(1) 指针碰撞(Bump the Pointer):假设Java堆中的内存是绝对规整的——所有用过的内存放一边,空闲的内存放内一边,中间放着一个指针作为分界点的指示器。分配内存即把指针向空闲的那边挪动一段与对象大小相等的距离;
(2) 空闲列表(Free List):假设Java堆中的内存是不规整的,已使用的内存和未使用的内存相互交错,这时,虚拟机就必须维护一个列表,记录哪些内存块是可用的,在分配内存的时候从列表中找到一块足够大的空间划分给对象实例,并更新列表上的记录;
选择哪种分配方式是由Java堆是否规整决定的,而Java堆是否规整则由所采用的垃圾收集器是否带有压缩整理功能决定,因此,在使用Serial、ParNew等带Compact过程的收集器时,系统采用的分配算法是指针碰撞,而使用CMS这种基于Mark-Sweep算法的收集器时,通常采用空闲列表;

如何保证划分内存是线程安全的:
(1) 方法之一是对分配内存的动作进行同步处理,虚拟机采用CAS配上失败重试的方式保证更新操作的原子性;
(2) 方法之二是把内存分配的动作按照线程划分在不同的空间中进行,即每个线程在Java堆中预先分配一小块内存,称为本地线程分配缓冲(Thread Local Allocation Buffer,TLAB),哪个线程需要分配内存,就在哪个线程的TLAB上分配,只有TLAB用完并分配新的TLAB时,才需要同步锁定;
虚拟机是否使用TLAB,可以通过-XX:+/-UseTLAB参数来设定;

将分配的内存空间设置为零值:内存分配完成后,虚拟机需要将分配到的内存空间都初始化为零值(不包括对象头),如果使用TLAB,这一工作也可以提前至分配TLAB时进行。这一步操作保证了对象的实例字段在Java中可以不赋初始值就直接使用,程序能访问这些字段的数据类型所对应的零值。

设置对象的头信息:设置这个对象是哪个类的实例,如何才能找到类的元数据信息、对象的哈希码、对象的GC分代年龄等信息。

 

Logo

华为开发者空间,是为全球开发者打造的专属开发空间,汇聚了华为优质开发资源及工具,致力于让每一位开发者拥有一台云主机,基于华为根生态开发、创新。

更多推荐