新建一个对象的步骤
一 虚拟机遇到一条new指令时,首先将检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并检查这个符号引用代表的类是否已经被加载,解析和初始化过。如果没有,那必须先执行相应的类加载过程。
二 类加载检查后,就是分配内存。分配内存根据不同的虚拟机内存回收策略,可以采用指针碰撞,空闲链表等方法划分空间。
除了划分空间,还有如何保证线程安全。线程安全有两种方案:1是对分配空间的动作进行同步处理,如cas失败重试保证原子操作;2是把内存分配的动作按照线程划分到不同的空间中进行(本地线程缓存区:TLAB)。
三 内存分配完成后,虚拟机需要将分配到的内存空间都初始化为0(不包括对象头),保证对象的实例字段在java代码中可以不赋初值就直接使用。
四 接下来,虚拟机要对对象进行必要的设置,例如这个对象是哪个类的实例,如何才能找到类的元数据信息,对象的哈希码,对象的gc分代年龄等信息。这些信息存放在对象头之中。
对象的内存布局
对象在内存中分为3块区域:对象头,实例数据和对齐填充
对象头
对象头分为两部分信息:第一部分用于存储对象自身的运行时数据,如哈希码,gc分代年龄,锁状态标识,线程持有的锁,偏向线程ID,偏向时间戳等。
第二部分时类型指针,即对象指向它的类元素据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。如果是一个java数组,还需要有一块用于纪录数组长度的数据。
实例数据
对象真正有效信息,程序代码中定义的各种类型的字段内容,无论是从父类继承下来的,还是在子类中定义的
对其填充
vm自动内存管理系统要求对象起始地址必须是8字节的整数倍,换句话说,就是对象的大小必须是8字节的整数倍。
对象的访问
java程序通过栈上的reference数据来操作堆上的具体对象。分为句柄访问和直接访问。
句柄访问相当于做了一层封装,在java堆中划分出一块来做为句柄池,然后句柄中包含了对象实例数据和类型数据的地址。reference指向句柄,句柄在指向对象。
OutOfMemoryError异常
- java堆溢出:
java.lang.OutOfMemoryError: Java heap space
通过MAT来查看,是哪些内存多了,并且无用 - 虚拟机栈和本地方法溢出
如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError异常
如果虚拟机就在扩展栈时无法申请到足够的空间,抛出OutOfMemoryError异常。 - 方法区和运行时常量池溢出
java.lang.OutOfMemoryError: PermGen space 说明运行时常量池数据方法区(永久代)