Java:JVM垃圾回收机制[通俗易懂]

Java:JVM垃圾回收机制[通俗易懂]JVM垃圾回收机制提到Java垃圾回收机制就不得不提到一个方法:system.gc()用于调用垃圾收集器,在调用时垃圾收集器将运行以回收未使用的内存空间,它将尝试释放被丢弃对象所占用的空间。作为程序员有必要了解gc方法,这也是在面试中经常会被问及的问题:我们从三个方面来理解gc:1.JVM如何确定哪些空间能被回收?2.JVM会在什么时候进行垃圾清除的动作?3.JVM如何清除垃圾的?1.JVM如…

大家好,又见面了,我是你们的朋友全栈君。

JVM垃圾回收机制

提到Java垃圾回收机制就不得不提到一个方法:
system.gc() 用于调用垃圾收集器,在调用时垃圾收集器将运行以回收未使用的内存空间,它将尝试释放被丢弃对象所占用的空间。

作为程序员有必要了解gc方法,这也是在面试中经常会被问及的问题:
我们从三个方面来理解gc:
1.JVM如何确定哪些空间能被回收?
2.JVM会在什么时候进行垃圾清除的动作?
3.JVM如何清除垃圾的?


1.JVM如何确定哪些空间能被回收
通过两个算法:
  • 引用计数算法:
     简单的来说就是判断对象的引用数量。实现方式:给对象共添加一个引用计数器,每当有引用对他进行引用时,计数器的值就加1,当引用失效,也就是不在执行此对象是,他的计数器的值就减1,若某一个对象的计数器的值为0,那么表示这个对象没有人对他进行引用,也就是意味着是一个失效的垃圾对象,就会被gc进行回收。
     但是这种简单的算法在当前的jvm中并没有采用,原因是他并不能解决对象之间循环引用的问题。
     假设有AB两个对象之间互相引用,也就是说A对象中的一个属性是BB中的一个属性时A,这种情况下由于他们的相互引用,从而是垃圾回收机制无法识别。
  • 可达性分析算法:
    因为引用计数法的缺点有引入了可达性分析算法,通过判断对象的引用链是否可达来决定对象是否可以被回收。可达性分析算法是从离散数学中的图论引入的,程序把所有的引用关系看作一张图,通过一系列的名为GC Roots的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链。当一个对象到 GC Roots 没有任何引用链相连(就是从 GC Roots 到这个对象不可达)时,则证明此对象是不可用的。

2.JVM会在什么时候进行垃圾清除的动作
  • 会在cpu空闲的时候自动进行回收
  • 在堆内存存储满了之后
  • 主动调用System.gc()后尝试进行回收(不一定成功)

3.JVM如何清除垃圾的?
通过四个算法:
  1. 标记-清除算法:

标记
所有需要回收的对象,然后
清除
标记的对象
缺点:会产生
空间碎片
,资源浪费
  1. 复制算法

内存分为两块
,当使用的一块满了就将
存活对象
复制到另一块中,再将使用的那一块清空
缺点:内存缩小约为原来的一半

  1. 标记-整理算法

标记
所有需要回收的对象,然后
清除
标记的对象,再将剩下的
存活对象整理
,避免了空间碎片的产生

  1. 分代收集算法
分代收集算法是比较智能的垃圾回收算法,也是现在JVM使用最多的算法,他本身其实并不算是一种算法,而是会在具体的场景上选择上面三种方法来进行垃圾回收
“代”指的是
新生代、老年代、永久代
【新生代】
按8:1:1分为
eden、survivorl0、survivor1
三个区域
一般情况下所有新生成的对象都存放于新生代
在新生代中,每次垃圾收集时都发现有
大批对象死去
,只有少量存活,那就选用复制算法。只需要付出少量
存活对象
的复制成本就可以完成收集。
【老年代】
老年代中存放的一般都是生存周期比较长的对象
老年代中因为
对象存活率高
、没有额外空间对他进行分配担保,就必须用标记

清除或者标记

整理
【永久代】
主要用来存放静态文件
【注意】在jdk8的时候Java废弃了永久代,同时提供了与永久代类似的叫做“元空间”的技术。元空间的本质和永久代类似。不过元空间与永久代之间最大的区别在于:元空间并不在虚拟机中,而是使用本地内存。也就是不局限于
jvm
可以使用系统的内存。理论上取决于
32

/
64
位系统可虚拟的内存大小。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/136666.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • JAVA/request.getParameterValues

    JAVA/request.getParameterValues1比较request.getParameterValues与request.getParameterrequest.getParameterValues(Stringname)是获得如checkbox类(名字相同,但值有多个)的数据。request.getParameter(Stringname)是获得对应名字的值,如果有重复的名,则返回第一个值。例如:reque

    2022年7月22日
    13
  • 括号匹配算法「建议收藏」

    概述​括号匹配在很多字符串处理的场景中时常被用到,诸如各大IDE括号不匹配的错误提示,编译器编译时检查应该成对出现的括号是否符合要求等,在这里我们就直接使用一种比较常规,但效率不差的方法去解决括号匹配的问题就行了。栈方法匹配问题​为了方便描述,对于需要做匹配的两个符号,比如’(‘和’)’,前者可称为左侧符号,后者可称为右侧符号。在做符号匹配时,如果以左侧符号

    2022年4月13日
    63
  • JVM基础和调优[通俗易懂]

    JVM基础和调优[通俗易懂]CMS和G1回收器,G1的并发标记周期,为什么要JVM调优?减少GC,特别是FullGC对象优先在堆的Eden区分配大对象直接进入老年代长期存活的对象将直接进入老年代.当Eden区没有足够的空间进行分配时,

    2022年6月9日
    34
  • 伪代码的写法[通俗易懂]

    伪代码的写法[通俗易懂]伪代码(Pseudocode)是一种算法描述语言。使用伪代码的目的是为了使被描述的算法可以容易地以任何一种编程语言(Pascal,C,Java,etc)实现。因此,伪代码必须结构清晰、代码简单、可读性好,并且类似自然语言。介于自然语言与编程语言之间。  它以编程语言的书写形式指明算法的职能。相比于程序语言(例如Java,C++,C,Dephi等等)它更类似自然语言。它是半角式化、不标准的语言

    2022年6月3日
    134
  • StringBuilder和StringBuffer的区别

    StringBuilder和StringBuffer的区别StringBuilder和StringBuffer的区别在哪里?当接触这个问题的时候,我们可能第一反应就是,StringBuilder是线程不安全的,StringBuffer是线程安全的为什么StringBuilder是线程不安全,StringBuffer是线程安全?针对这个问题,大部分的人可能就无言以对了,我们只知道StringBuilder是线程不安全的,StringBuffer…

    2022年6月28日
    25
  • dede使用方法—-调用导航

    dede使用方法—-调用导航

    2021年9月19日
    55

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号