创建对象在堆区如何分配内存

            Java程序在运行时,在虚拟机的自动内存管理机制下,不需要去关注内存泄漏和内存溢出的问题,但是如果发生了StackOverFlowError OutOfMemoryError,我们就得知道这是虚拟机 栈或者是内存管理出了问题。所以我们必须了解一下Java内存模型(JVM)。

1.程序运行时的内存划分

  • JVM虚拟机在执行 Java 程序的过程中会把它管理的内存划分成若干个不同的数据区域。
  • JDK 1.8之前分为:线程共享Heap堆区、Method Area方法区)、线程私有(虚拟机栈、本地方法栈、程序计数器)
  • JDK 1.8以后分为:线程共享Heap堆区、MetaSpace 元空间)、线程私有(虚拟机栈、本地方法栈、程序计数器。 

 

 2.堆(Heap)

      Heap堆区,用于存放对象实例和数组。

  Heap堆是JVM 所管理的内存中最大的一块区域,被所有线程共享的一块内存区域。堆区中存放对象实例,“几乎”所有的对象实例以及数组都在这里分配内存

新生代、老年代

     Heap堆是垃圾收集器GCGarbage Collected管理的主要区域,因此堆区也被称作GC 堆(Garbage Collected Heap。从垃圾回收的角度,由于现在收集器基本都采用分代垃圾收集算法,所以 JVM中的堆区往往进行分代划分,例如:新生代 老年代目的是更好地回收内存,或者更快地分配内存。

 

3.创建一个新对象,在堆中的分配内存。

大部分情况下,对象会在 Eden 区生成,当 Eden 区装填满的时候,会触发 Young Garbage Collection,即 YGC垃圾回收的时候,在 Eden 区实现清除策略,没有被引用的对象则直接回收。

依然存活的对象会被移送到 Survivor 区。Survivor 区分为 s0 s1 两块内存区域。每次 YGC的时候,它们将存活的对象复制到未使用的Survivor 空间(s0s1),然后将当前正在使用的空间完全清除,交换两块空间的使用状态。每次交换时,对象的年龄会加+1

如果 YGC 要移送的对象大于 Survivor 区容量的上限,则直接移交给老年代。一个对象也不可能永远呆在新生代,在 JVM 中 一个对象从新生代晋升到老年代的阈值默认值是 15,可以在 Survivor区交换 14 次之后,晋升至老年代。

 堆区最容易出现的问题就是OutOfMemoryError错误,会有俩种表现形式:

(1)GC Overhead Limit Exceeded:当JVM花太多时间执行垃圾回收,并且只能回收很少的堆空间,就会发生此错误。

(2)Java Heap space:假如在创建新的对象时,堆内存中的空间不足以存放新创建的对象,就会引发此错误。

  • 10
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 6
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

#0000FF格子衫

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值