0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Java所承诺的自动内存管理

存储界 来源:未知 作者:李倩 2018-04-29 15:16 次阅读

Java所承诺的自动内存管理主要是针对对象内存的回收和对象内存的分配。

在Java虚拟机的五块内存空间中,程序计数器、Java虚拟机栈、本地方法栈内存的分配和回收都具有确定性,一般在编译阶段就能确定需要分配的内存大小,并且由于都是线程私有,因此它们的内存空间都随着线程的创建而创建,线程的结束而回收。也就是这三个区域的内存分配和回收都具有确定性,垃圾回收器不需要在这里花费太大的精力。

而Java虚拟机中的方法区因为是用来存储类信息、常量、静态变量,这些数据的变动性较小,因此不是Java内存管理重点需要关注的区域。

而对于堆,所有线程共享,所有的对象都需要在堆中创建和回收。虽然每个对象的大小在类加载的时候就能确定,但对象的数量只有在程序运行期间才能确定,因此堆中内存的分配具有较大的不确定性。此外,对象的生命周期长短不一,因此需要针对不同生命周期的对象采用不同的内存回收算法,增加了内存回收的复杂性。

综上所述:Java自动内存管理最核心的功能是堆内存中对象的分配与回收。

对象优先在Eden区中分配

目前主流的垃圾收集器都会采用分代回收算法,因此需要将堆内存分为新生代和老年代。

在新生代中为了防止内存碎片问题,因此垃圾收集器一般都选用“复制”算法。因此,堆内存的新生代被进一步分为:Eden区+Survior1区+Survior2区。

每次创建对象时,首先会在Eden区中分配。

若Eden区已满,则在Survior1区中分配。

若Eden区+Survior1区剩余内存太少,导致对象无法放入该区域时,就会启用“分配担保”,将当前Eden区+Survior1区中的对象转移到老年代中,然后再将新对象存入Eden区。

大对象直接进入老年代

所谓“大对象”就是指一个占用大量连续存储空间的对象,如数组。

当发现一个大对象在Eden区+Survior1区中存不下的时候就需要分配担保机制把当前Eden区+Survior1区的所有对象都复制到老年代中去。

我们知道,一个大对象能够存入Eden区+Survior1区的概率比较小,发生分配担保的概率比较大,而分配担保需要涉及到大量的复制,就会造成效率低下。

因此,对于大对象我们直接把他放到老年代中去,从而就能避免大量的复制操作。

那么,什么样的对象才是“大对象”呢?

通过-XX:PretrnureSizeThreshold参数设置大对象

该参数用于设置大小超过该参数的对象被认为是“大对象”,直接进入老年代。

注意:该参数只对Serial和ParNew收集器有效。

生命周期较长的对象进入老年代

老年代用于存储生命周期较长的对象,那么我们如何判断一个对象的年龄呢?

新生代中的每个对象都有一个年龄计数器,当新生代发生一次MinorGC后,存活下来的对象的年龄就加一,当年龄超过一定值时,就将超过该值的所有对象转移到老年代中去。

使用-XXMaxTenuringThreshold设置新生代的最大年龄

设置该参数后,只要超过该参数的新生代对象都会被转移到老年代中去。

相同年龄的对象内存超过Survior内存一半的对象进入老年代

如果当前新生代的Survior中,年龄相同的对象的内存空间总和超过了Survior内存空间的一半,那么所有年龄相同的对象和超过该年龄的对象都被转移到老年代中去。无需等到对象的年龄超过MaxTenuringThreshold才被转移到老年代中去。

“分配担保”策略详解

当垃圾收集器准备要在新生代发起一次MinorGC时,首先会检查“老年代中最大的连续空闲区域的大小 是否大于 新生代中所有对象的大小?”,也就是老年代中目前能够将新生代中所有对象全部装下?

若老年代能够装下新生代中所有的对象,那么此时进行MinorGC没有任何风险,然后就进行MinorGC。

若老年代无法装下新生代中所有的对象,那么此时进行MinorGC是有风险的,垃圾收集器会进行一次预测:根据以往MinorGC过后存活对象的平均数来预测这次MinorGC后存活对象的平均数。

如果以往存活对象的平均数小于当前老年代最大的连续空闲空间,那么就进行MinorGC,虽然此次MinorGC是有风险的。

如果以往存活对象的平均数大于当前老年代最大的连续空闲空间,那么就对老年代进行一次Full GC,通过清除老年代中废弃数据来扩大老年代空闲空间,以便给新生代作担保。

这个过程就是分配担保。

注意:

1. 分配担保是老年代为新生代作担保;

2. 新生代中使用“复制”算法实现垃圾回收,老年代中使用“标记-清除”或“标记-整理”算法实现垃圾回收,只有使用“复制”算法的区域才需要分配担保,因此新生代需要分配担保,而老年代不需要分配担保。

名词解释:

GC:垃圾收集器

Minor GC:新生代GC,指发生在新生代的垃圾收集动作,所有的Minor GC都会触发全世界的暂停(stop-the-world),停止应用程序的线程,不过这个过程非常短暂。

Major GC/Full GC:老年代GC,指发生在老年代的GC。

JVM:Java Virtual Machine(Java虚拟机)的缩写。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 内存
    +关注

    关注

    8

    文章

    2737

    浏览量

    72613
  • JAVA
    +关注

    关注

    19

    文章

    2903

    浏览量

    102867

原文标题:JVM学习,对象内存的分配策略

文章出处:【微信号:cunchujie,微信公众号:存储界】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    JAVA和C++区别

    指针来直接访问内存无指针,并且增添了自动内存管理功能,从而有效地防止了c/c++语言中指针操作失误,如野指针造成的系统崩溃。但也不是说
    发表于 04-11 15:19

    JAVA和C++区别

    来直接访问内存无指针,并且增添了自动内存管理功能,从而有效地防止了c/c++语言中指针操作失误,如野指针造成的系统崩溃。但也不是说
    发表于 10-10 14:50

    具有自动内存管理功能的USB DMA

    我找不到任何有意义的文件,用于USB DMA W/自动内存管理。有一个例子被张贴在论坛,但是它不显示任何数据传输,所以大多是无用的,一些正确的问题被问在论坛上却没有回答。我不知道这种模式是不被任何人
    发表于 09-05 09:47

    java线程内存模型

    一、Java内存模型 按照官方的说法:Java 虚拟机具有一个堆,堆是运行时数据区域,所有类实例和数组的内存均从此处分配。 JVM主要管理
    发表于 09-27 10:55 0次下载
    <b class='flag-5'>java</b>线程<b class='flag-5'>内存</b>模型

    实例分析内存回收机制在java中的应用

    Java中,它的内存管理包括两方面:内存分配(创建Java对象的时候)和内存回收,这两方面工作
    发表于 09-27 15:10 0次下载
    实例分析<b class='flag-5'>内存</b>回收机制在<b class='flag-5'>java</b>中的应用

    Java内存模型及原理分析

    一、Java内存模型 按照官方的说法:Java 虚拟机具有一个堆,堆是运行时数据区域,所有类实例和数组的内存均从此处分配。 JVM主要管理
    发表于 09-28 11:49 0次下载
    <b class='flag-5'>Java</b><b class='flag-5'>内存</b>模型及原理分析

    Java内存区域分配、Java虚拟机栈、对象的访问方式和GC

    对于Java程序员来说,在虚拟机的自动内存管理机制的帮助下,不再需要为每一个new操作去写配对的delete/free代码,而且不容易出现 内存
    发表于 12-11 16:28 2134次阅读
    <b class='flag-5'>Java</b><b class='flag-5'>内存</b>区域分配、<b class='flag-5'>Java</b>虚拟机栈、对象的访问方式和GC

    C++内存管理技术的详细资料说明

    泄漏几乎在每个C++程序中都会发生,因此要想成为C++高手,内存管理一关是必须要过的,除非放弃C++,转到Java或者.NET,他们的内存管理
    发表于 03-14 08:00 4次下载

    Android内存泄漏问题如何优化?

    作者:无名之辈FTER 来源:CSDN博客 众所周知,Java因其拥有独特的虚拟机(JVM)设计,使其成为一门跨平台、内存自动管理的高级开发语言。所谓跨平台,即一次编译,多次运行,从而
    的头像 发表于 09-11 15:55 1569次阅读

    详解Java虚拟机的JVM内存布局

    JVM内存布局规定了Java在运行过程中内存申请、分配、管理的策略,保证了JVM的稳定高效运行。不同的JVM对于内存的划分方式和
    的头像 发表于 07-13 09:52 299次阅读
    详解<b class='flag-5'>Java</b>虚拟机的JVM<b class='flag-5'>内存</b>布局

    java内存溢出的几种原因和解决办法

    Java是一种使用垃圾回收机制的编程语言,由于自动内存管理机制的存在,Java程序中发生内存溢出
    的头像 发表于 11-23 14:44 2022次阅读

    java内存溢出排查方法

    Java内存溢出(Memory overflow)是指Java虚拟机(JVM)中的堆内存无法满足对象分配的需求,导致程序抛出OutOfMemoryError异常。
    的头像 发表于 11-23 14:46 549次阅读

    如何查看java程序的内存分布

    要查看Java程序的内存分布,首先需要了解Java程序运行时的内存模型。 Java程序的内存分布
    的头像 发表于 11-23 14:47 458次阅读

    jvm管理内存包括哪几个运行时数据内存

    JVM(Java虚拟机)是Java程序的运行环境,它提供了内存管理机制来管理Java程序所需的运
    的头像 发表于 12-05 14:09 170次阅读

    java虚拟机内存包括远空间内存

    将详细介绍JVM内存的各个部分及其作用。 Java堆(Heap) Java堆是JVM管理的最大一块内存区域,用于存放
    的头像 发表于 12-05 14:15 164次阅读