JVM学习笔记day5

对象访问

  • 看如下这行代码
 Object object = new Object();

"Object object"这部分的语义将会反映到Java栈的本地变量表中,作为一个reference类型数据出现。而"new Object()"这部分语义将会反映到Java堆中,形成一块存储了Object类型所有实例数据值的结构化内存,根据具体类型以及虚拟机实现的对象内存布局的不同,这块内存的长度是不固定的。另外,在Java堆中还必须包含能查到此对象类型的数据(如对象类型、父类、实现的接口、方法等)的地址信息,这些类型数据则存储在方法区中。

  • 由于reference类型在Java虚拟机规范里面只规定了一个指向对象的引用,并没有定义这个引用应该通过哪种方式去定位,以及访问到Java堆中的对象的具体位置,因此不同虚拟机实现的对象访问方式会有所不同,主流的访问方式有两种:使用句柄直接指针
    • 如果是使用句柄访问方式,Java堆中将会划分出一块内存来作为句柄池,reference中存储的就是对象的句柄地址,而句柄中包含了对象实例数据和类型数据各自的具体地址信息,如图所示
      在这里插入图片描述
    • 如果使用直接指针访问方式,Java堆对象的布局中就必须考虑如何放置访问类型数据的相关信息,reference中直接存储的就是对象地址
      在这里插入图片描述
      这两种对象的访问方式各有优势,使用句柄访问方式的最大好处就是reference中存储的是稳定的句柄地址,在对象被移动时只会改变句柄中的实例数据指针,而reference本身不需要被修改。
      使用直接指针访问方式的最大好处是速度更快,节省了一次指针定位的时间开销,由于对象的访问在Java中非常频繁,因此这类开销积少成多后也是一项非常可观的执行成本

OutOfMemoryError异常

  • Java堆溢出
    • Java堆用于存储对象实例,我们只要不断地创建对象,并且保证GC Roots到对象之间有可达路径来避免垃圾回收机制清除这些对象,就会在对象数量到达最大堆的容量限制后产生内存溢出异常。

设置JVM参数:
在这里插入图片描述

   package com.ginger;

import java.util.ArrayList;
import java.util.List;

/**
* @author ginger
* @date 2019/11/20 14:49
*/
public class Test11 {
   static class OOMObject{

   }

   public static void main(String[] args) {
       List<OOMObject> list = new ArrayList<>();
       while (true){
           list.add(new OOMObject());
       }
   }
}

运行结果:
在这里插入图片描述

  • 虚拟机栈和本地方法栈溢出
    • 由于在HotSpot虚拟机中并不区分虚拟机栈和本地方法栈,因此对于HotSpot来说,-Xoss参数(设置本地方法栈大小)虽然存在,但实际上是无效的,栈容量只由-Xss参数设定。关于虚拟机栈和本地方法栈,在Java虚拟机规范中描述了两种异常:
      • 如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError异常
      • 如果虚拟机在拓展栈时无法申请到足够的内存空间,则抛出OutOfMemoryError异常
package com.ginger;

/**
* @author ginger
* @date 2019/11/20 16:57
*/
public class Test12 {
   private int stackLength = 1;
   public void stackLeak(){
       stackLength++;
       stackLeak();
   }

   public static void main(String[] args){
       Test12 oom = new Test12();
       try{
           oom.stackLeak();
       }catch (Throwable e){
           System.out.println("stack length:"+oom.stackLength);
           throw e;
       }
   }
}

运行结果:
在这里插入图片描述

  • 运行时常量池溢出
    -如果要向运行时常量池中添加内容,最简单的做法就是使用String,intern()这个Native方法。
package com.ginger;

import java.util.ArrayList;
import java.util.List;

/**
* @author ginger
* @date 2019/11/20 17:24
*/
public class test13 {
  public static void main(String[] args) {
      List<String> list = new ArrayList<>();
      int i = 0;
      while(true){
          list.add(String.valueOf(i++).intern());
      }
  }
}

在这里插入图片描述

  • 方法区溢出
    • 方法区用于存放Class的相关信息,如类名、访问修饰符、常量池、字段描述、方法描述等。对于这个区域的测试,基本的思路是运行时产生大量的类去填满方法区,直到溢出。
      在这里插入图片描述
      在这里插入图片描述
      方法区溢出也是一种常见的内存溢出异常,一个类如果要被垃圾回收器回收掉,判定条件是非常苛刻的。在经常动态生成大量Class的应用中,需要特别注意类的回收状况
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值