你应该这样理解JVM内存管理
在进行Java程序设计时,一般不涉及内存的分配和内存回收的相关代码,此处引用一句话: Java和C++之间存在一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外的人想进去,墙里面的人想出来 ,个人从这两句话中,捕获到了 两个点 。
- java的自动内存管理机制,极大的节省了开发人员的精力,避免了易错且复杂的内存管理设计,相对于手动的内存管理这是极大的飞跃。
- java自动内存管理机制,其不能根据具体的场景提供最优的内存管理,其只提供普适的内存管理机制。想比于C++的手动内存管理,灵活性不够,存在制约系统性能的瓶颈。
第二点是我们深入了解JVM内存管理机制的意义,通过对原理的把握,在指定的场景下设计JVM最优的内存管理策略,本文内容组织结构如下:
- JVM内存分配
- JVM内存回收
2. JVM内存分配
ClassExample refereenceExample = new ClassExample ();复制代码
上述代码,如果粗略划分的话,可以划分为两个过程:
- 在堆区分配对象内存
- 将对象内存地址赋值给对象引用 上述两个过程基本就是内存分配中,比较重要的两个知识点了, 内存分配策略和对象引用 #####2.1 堆区布局 对象分配在堆上这是毫无疑问的,如果在往下细分的话,那么堆区的内存布局还是挺有讲究的,大致可以分为如下布局:
- 新生代:从名字可见一般,新生代区域中存放的对象一般是两种: 刚刚被new出来的对象,或者经历内存回收次数不多的对象 。
- 老年代:老年代从名字也可见一般,一般存放两种对象: 占用内存比较多的大对象和经历过多次内存回收过程的对象
- 永久代:永久代的内存就比较固定,每个类的class对象,一般存放在永久代当中。
上述JVM堆区中的内存布局代表的是 逻辑视图 ,并不是实际的物理布局,实际上,JVM了提供多种不同的内存分配和回收的策略,每种策略抽闲出逻辑视图都会有细微的差别,但是上述逻辑视图可以说是所有逻辑视图的根视图
2.2 内存分配一般过程
在图中有几个重要的概念,需要着重强调:
- 对象大小阈值设定 ,在JVM中可以通过设PretenureSizeThreshold这个参数,指定该阈值大小,如果待分配对象大小超过该阈值,尝试在老年代中开辟内存空间存储对象,否则尝试在新生代中开辟内存空间存储对象
- MinorGC ,这是JVM中一次内存回收过程,内存回收过程又称垃圾回收过程, 这次内存回收是由新生代中内存空间不足引起的,主要对新生代中的内存对象进行回收
- FullGC ,这也是JVM中的一次内存回收过程, 这次内存回收过程是有老年代中内存空间不足造成的,主要针对所有堆区中的内存对象进行回收,包括新生代,老年代,以及永久代
- 对象年龄 ,经过一次内存回收后依然存活的对象,其年龄会加1。当对象年龄超过一个指定阈值后,其由新生代转向老年代存储。这个对象年龄的阈值,同样可以通过设置JVM的MaxTenuringThreshold参数进行指定
2.3对象引用
JVM内存区域的布局详情 中介绍了对象引用相关的内容
3. JVM内存回收
从程序员角度来看,内存回收的过程是透明,具体细节都对程序员屏蔽了。 JVM内存区域的布局详情 ,仔细的介绍了JVM中的内存模型以及各个内存区存储的数据类型。
内存回收主要针对的内存区域主要是堆区和方法区,在上文中谈及了MinorGC以及FullGC。MinorGC主要是针对堆区进行内存回收,FullGC除了对堆区进行回收,对方法区也进行回收,是相对重量级的回收动作。在内存回收这个章节中,主要从以下三个方面阐述: 1. 什么样的对象可回收?解决方法:可达性分析 2. 如何回收内存? 解决方法:标记-清除算法,标记-复制算法,标记-整理算法等内存回收算法 3.什么时候回收合适? 解决方法:安全点和安全区域3.1 可达性分析
在上图中,一共有七个对象,箭头的走向代表引用关系、对象1引用对象2,对象2引用对象3。 对象1是根节点(GCROOT),其他的是非根节点(GCROOT)。 **可达性分析是指:不能被GCROOT节点通过引用链达到的节点,将被列入拟回收对象范围。对象6和对象7就不能被任何一个GCRoot节点所引用,是目标回收对象。**JVM认为这些对象不具备使用价值,可以将其进行内存回收。 **根节点(GCROOT)**是上文中提出的重要概念,一般来说,如下对象可作为根节点对象: (1)JVM虚拟机栈中引用的对象 (2)方法区中类静态属性引用对象 (3)方法区中类常量引用对象 (4)本机方法栈中引用对象 可达性分析是进行内存回收的判定条件,在可达性分析之后,确定哪些对象是回收目标。除了可达性分析之外, 引用计数法同样可用来进行判断内存回收目标对象,但是其无法解决循环引用问题。主流的JVM都采用可达性分析。
3.2内存回收算法
通过可达性分析之后,将确定哪些对象是回收目标,接着内存回收算法将执行具体的回收细节。下图是内存区域中三个状态,空闲内存是未使用内存,目标回收内存是可达性分析之后可回收内存,已占内存是不需要回收内存。
- 标记-清除算法 标记-清除算法是比较直白的内存回收算法,其直接释放目标回收内存,没有其他任何附加操作。该算法执行后的结果如下:
- 标记-清除算法,逻辑简单,易于理解,但是其有个致命的缺点就是其极易产生内存碎片,在分配大对象时,不能有足够连续的内存空间而造成频繁的GC。
- 标记-复制算法 标记-复制算法是针对内存碎片问题提出的一种算法。
- 在标记-复制算法的设定中,始终有块空白内存区域,如上图长内存区域二。在内存回收时,将内存区域一种存活对象全部按序复制到内存区域二中后再对内存区域一中所有对象进行回收。通过这种做法可以极大的避免内存碎片,但是空白内存区域将始终闲置,内存利用率不高。 在上文中提及将堆区的内存分为两个surivior区以及Eden区就是采用标记-复制算法。Eden区和surivior之间大小比例分配默认是8:1:1.
- 标记-整理算法 标记-整理算法,同样是为解决内存碎片提出的内存回收算法,其在标记-清楚算法的基础上增加的整理功能,对内存对象进行拷贝移动,以保证空间内存空间连续。
- 标记-整理算法,能够克服内存碎皮,且不需要额外内存,但是其拷贝和移动对象的时间消耗较高。该算法常用在老年代对象回收中。
3.3 什么时候回收内存合适
通过可达性分析,可知哪些对象是需要回收的对象,在这过程中需要枚举根节点,这是一个耗时操作,为了保证内存回收的顺利进行,必须保证引用关系的一致性,即在内存回收过程中对象引用关系不会发生变化。只有在这些地方进行内存回收才是安全点,因此JVM引入了 安全点 的概念,安全点处的对象引用关系不会改变,适合内存回收。 安全区域 是对 安全点概念的扩充 ,指在这一段区域内对象引用关系具有一致性,能进行内存回收。
喜欢的朋友点点关注
你应该这样理解JVM内存管理的更多相关文章
- Java之美[从菜鸟到高手演变]之JVM内存管理及垃圾回收
很多Java面试的时候,都会问到有关Java垃圾回收的问题,提到垃圾回收肯定要涉及到JVM内存管理机制,Java语言的执行效率一直被C.C++程序员所嘲笑,其实,事实就是这样,Java在执行效率方面确 ...
- JVM内存管理及垃圾回收
一.JVM内存的构 Java虚拟机会将内存分为几个不同的管理区,这些区域各自有各自的用途,根据不同的特点,承担不同的任务以及在垃圾回收时运用不同的算法.总体分为下面几个部分: 程序计数器(Progra ...
- java虚拟机学习-JVM内存管理:深入垃圾收集器与内存分配策略(4)
Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人却想出来. 概述: 说起垃圾收集(Garbage Collection,下文简称GC),大部分人都把这项 ...
- java虚拟机学习-JVM内存管理:深入Java内存区域与OOM(3)
概述 Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人却想出来. 对于从事C.C++程序开发的开发人员来说,在内存管理领域,他们即是拥有最高权力的皇帝又 ...
- Java的内存 -JVM 内存管理
一.综述 如果你学过C或者C++,那么你应该感受过它们对内存那种强大的掌控力.但是强大的能力往往需要更强大的控制力才能保证能力不被滥用,如果滥用C/C++的内存管理那么很容易出现指针满天飞的情况,不出 ...
- 走进JVM【二】理解JVM内存区域
引言 对于C++程序员,内存分配与回收的处理一直是令人头疼的问题.Java由于自身的自动内存管理机制,使得管理内存变得非常轻松,不容易出现内存泄漏,溢出的问题. 不容易不代表不会出现问题,一旦内存泄漏 ...
- JVM内存管理及垃圾回收【转】
很多Java面试的时候,都会问到有关Java垃圾回收的问题,提到垃圾回收肯定要涉及到JVM内存管理机制,Java语言的执行效率一直被C.C++程序员所嘲笑,其实,事实就是这样,Java在执行效率方面确 ...
- JVM内存管理及GC机制
一.概述 Java GC(Garbage Collection,垃圾收集,垃圾回收)机制,是Java与C++/C的主要区别之一,作为Java开发者,一般不需要专门编写内存回收和垃圾清理代码,对内存泄露 ...
- JVM内存管理:深入Java内存区域与OOM
Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人却想出来. 概述: 对于从事C.C++程序开发的开发人员来说,在内存管理领域,他们即是拥有最高权力的皇帝 ...
随机推荐
- python3.5学习第一章
在python中,变量赋值或者其他语句后不需要加”:“ python中,缩进特别重要! 条件语句和循环语句,要加”:“ 一.变量赋值 name = "cecelia"name2 = ...
- Python量化库大全
https://zhuanlan.zhihu.com/p/26983703?utm_source=wechat_session&utm_medium=social 这个网址上详细介绍了,做量化 ...
- 子数组最小值的总和 Sum of Subarray Minimums
2018-09-27 23:33:49 问题描述: 问题求解: 方法一.DP(MLE) 动态规划的想法应该是比较容易想到的解法了,因为非常的直观,但是本题的数据规模还是比较大的,如果直接使用动态规划, ...
- 非常好的 gdb tui 的文章
http://beej.us/guide/bggdb/ Help Commands help command Get help on a certain command apropos keyword ...
- English trip V1 - 24. Accommodations Teacher:Maple Key: make suggestions 提出建议
In this lesson you will learn to make suggestions. 在本课程中,您将学习如何提出建议. 课上内容(Lesson) Which place would ...
- 使用Service组件实现简单的音乐播放器功能 --Android基础
1.本例利用Service实现简单的音乐播放功能,下面是效果图.(点击开始播放开启服务,音乐播放,点击“停止播放”关闭服务,音乐停止播放.) 2.核心代码: MusicService.java: pa ...
- ionic this.navCtrl.push()和this.navCtrl.pop()
在ionic中,this.navCtrl.push()和this.navCtrl.pop()都是进行页面跳转,但是用法又有区别 例如有A B C三个页面,三个页面都是使用this.navCtrl. ...
- 【异常及源码分析】org.mybatis.spring.MyBatisSystemException: nested exception is org.apache.ibatis.type.TypeException: Could not set parameters for mapping: ParameterMapping
一.异常出现的场景 1)异常出现的SQL @Select("SELECT\n" + " id,discount_type ,min_charge, ${cardFee} ...
- New Roads CodeForces - 746G (树,构造)
大意:构造n结点树, 高度$i$的结点有$a_i$个, 且叶子有k个. 先确定主链, 然后贪心放其余节点. #include <iostream> #include <algorit ...
- Robbers' watch CodeForces - 685A (暴力)
大意: 一天n小时, m分钟, 表以7进制显示, 求表显示数字不同的方案数 注意到小时和分钟部分总长不超过7, 可以直接暴力枚举. 关键要特判0, 0的位数要当做1来处理 #include <i ...