昨天我们用比较精简的文字讲了 Java 虚拟机结构,没看过的可以直接从这里查看:

每日一问:你了解 Java 虚拟机结构么?

今天我们必须来看看 Java 虚拟机的垃圾回收算法是怎样的。不过在开始之前,我们一定得确定哪些是活着的对象,又有哪些是可以进行回收的。

判断对象是否存活方式

引用计数算法

对应判断一个对象是否可以回收,我想引用计数一定是最容易被想到的算法了吧。给每个对象加一个引用计数器,每当有一个地方引用它时,计数器就加 1,引用失效后减 1,当对象的计数器为 0,则说明这个对象可以被回收了。这个算法非常简单,但存在一个非常大的弊端:一旦两个对象相互引用,这个算法就没辙了。

根搜索算法

Java 就是采用的根搜索算法进行判断对象是否存活。这个算法的思路是:通过一系列名为 "GC Roots" 的对象作为起始点,从这些结点开始向下搜索,当一个对象到 "GC Roots" 没有任何引用链相连的话,则证明这个对象是可以被回收的。在 Java 中,可以作为 "GC Roots" 的对象包括:

虚拟机栈中引用的对象;

方法区中的类静态属性和常量引用的对象;

本地方法栈中 JNI 引用的对象;

四种引用

在 JDK 1.2 之后,引用被分为了强引用、软引用、弱引用和虚引用四种,这四种引用强度依次逐渐减弱。

强引用

强引用在 Android 代码中普遍存在,只要强引用还在,垃圾回收器就不会回收掉被引用的对象,这就是为什么我们用内部类持有 Activity 实例会造成内存泄漏的根本原因。

软引用

软引用用来描述一些还有用,但非必需的对象,用 SoftReference 实现,被软引用关联的对象,在系统将要发生 OOM 之前,会把这些对象列进回收范围之中并进行第二次回收。软引用在 Android 中主要是用于做缓存,比如软引用缓存网络请求的图片。

弱引用

弱引用也是用来描述非必需对象的,但它的强度比软引用更弱,用 WeakReference 实现。被弱引用管理的对象只能生存到下一次垃圾收集发生之前。弱引用在 Android 中主要用于处理内存泄漏。

虚引用

虚引用其实没啥好说的,一个对象是否有虚引用的存在,完全不会对生存时间构成影响,也无法通过虚引用来取得一个对象实例。就目前为止,我还没有在 Android 开发中使用过它。

都有些什么垃圾回收算法

学习 Java 虚拟机的垃圾回收算法之前,我们必须来看看我们常见的几种垃圾回收算法的思想,并把它们的优劣进行一定的对比,这样一定才能让你理解更加深刻。

标记 - 清除算法

标记 - 清除算法应该是最简单基础的收集算法了,只需要标记需要回收的对象,标记完成后统一回收即可。但其有两个非常明显的弊端。

标记清除效率都不高;

标记清除后会产生大量不连续的内存碎片,导致程序以后需要较大对象时无法找到足够的连续内存而不得不提前触发另一次垃圾收集动作。

复制收集算法

复制算法主要是将可用内存划分为大小相等的两块,每次只使用其中的一块,当这一块内存用完,就将存活着的对象复制到另一块内存上去,然后把已使用过的内存空间一次性清理掉。复制回收算法能有效地避免内存碎片,但是算法需要把内存一分为二,导致内存使用率大大降低。

标记 - 整理算法

复制收集算法在对象存活率较高时就需要进行较多的复制操作,效率非很低。 效率会很低。标记-整理算法就解决了这样的问题,同样采用的是根搜索算法进行存活对象标记,但后续是将所有存活的对象都移动到内存的一端,然后清理掉端外界的对象。

分代收集算法

当前包括 Java 虚拟机在内的商业虚拟机都采用的是分代收集算法。这种算法其实就是根据对象的存活周期不同将内存划分为几块。一般把 Java 堆分为新生代和老年代,然后根据各个年代的特点采用最适合的收集算法。

Java 虚拟机的垃圾回收策略

前面说了 Java 虚拟机采用的是分代回收算法,该算法会根据各个年代的特点采用最适合的收集算法,我们就必须了解 Java 堆分的各个年代区域的特点。

JVM 中共分为三个代:新生代、老年代和持久代。其中持久代主要存放的是 Java 类的类信息,与垃圾收集要收集的 Java 对象关系不大。

新生代:所有新生成的对象首先都是放在新生代的,新生代采用复制回收算法。新生代的目标就是尽可能快速地收集掉那些生命周期短的对象。新生代按照 8:1 的比例分为一个 Eden 区和两个 Survivor 区。大部分对象在 Eden 区生成,当 Eden 区满时,还存活的对象将被复制到其中的一个 Survivor 区,当这个 Survivor 区满时,此区的存活对象将被复制到另外一个 Survivor 区,当另一个 Survivor 区也满了的时候,从第一个 Survivor 区复制过来的并且此时还存活的对象,将被复制到了「年老区 」。需要注意,Survivor 的两个区是对称的,没有任何的先后关系,所以同一个区中可能同时存在 Eden 复制过来的对象,和从前一个 Survivor 区复制过来的对象,而复制到年老区的只有从第一个 Survivor 区过来的对象,而且,Survivor 区总有一个是空的。

老年代:在新生代中经历了 N 次垃圾回收后仍然存活的对象,就会被放到老年代中,老年代采用标记整理回收算法。因此,可以认为老年代中存放的都是一些生命周期较长的对象。

持久代:用于存放静态文件,如 final 常量、static 常量、常量池等。持久代对垃圾回收没有显著影响,但有些应用可能动态生成或者调用一些 class。在这种时候需要设置一个比较大的持久代空间来存放这些运行过程中新增的类。

谈谈 Java 垃圾回收的触发条件

Java 垃圾回收包含两种类型:Scavenge GC 和 Full GC。

Scavenge GC:一般情况下,当新对象生成,并且在 Eden 申请空间失败的时候,就会触发 Scavenge GC,对 Eden 区进行 GC,清除非存活的对象,并且把尚且存活的对象移动到 Survivor 区,然后整理 Survivor 的两个区。这种方式的 GC 是对新生代的 Eden 区进行,不会影响到老年代。因为大部分对象都是从 Eden 区开始的,同时 Eden 区不会分配的很大,所以 Eden 区的 GC 会频繁进行。

Full GC:Full GC 将会对整个堆进行整理,包括新生代、老年代和持久代。Full GC 因为需要对整个堆进行回收,所以比 Scavenge GC 要慢,因此应该尽量减少 Full GC 的次数。在对 JVM 调优的过程中,很大一部分工作就是对 Full GC 的调节,有如下原因可能导致 Full GC:

老年代被写满;

持久代被写满;

System.gc() 被显式调用;

好了,这一篇文字比起前面的文字稍微多了一些,主要是知识关联性稍微大了一些,又不适合分开讲解,所以就只能这样了。

java虚拟机多久触发垃圾回收_每日一问:讲讲 Java 虚拟机的垃圾回收相关推荐

  1. java记录按钮点击次数_新手想问:JAVA怎么实现在规定时间记录点击按钮的次数...

    新手想问:JAVA怎么实现在规定时间记录点击按钮的次数 我估计问题应该是main方法里的循环没法实现,不知道怎么解决了. import java.awt.Color; import java.awt. ...

  2. java说影_面试被问到Java虚拟机,用这篇文章怼过去

    Jvm内存结构 Jvm内存结构,一般是面试官对Java虚拟机这块考察的第一问.讲真,还没背会,自己罚自己面壁思过. Java虚拟机的内存结构一般可以从线程共有和线程私有两部分起头作答,然后再详细说明各 ...

  3. Java好还是网优好_大神告诉你|Java好还是Python好?

    在进入主题之前,小智先分享几条大神的人生警句.醒世明言: 初学者才争论语言,我们只看心情. 还有: 选择入门语言最重要的是 "三观"相似 "臭味"相投 以及大家 ...

  4. java类sample是公共的_应在名samle.java的文件_Andoid NDK编程 1 - 注册native函数

    打算对Android的NDK的开发做一总结,首先是JNI部分,接下来是NDK的内容.今天首先介绍一下JNI的第一部分:注册native函数. 当java代码中执行native的代码时候,首先是通过一定 ...

  5. java里class有什么用_安装JDK时的java和javac命令有什么用?

    相信大家都装过JDK吧,在装JDK时我们会在命令提示符窗口中通过java和javac两个命令来检测我们的环境变量是否配置正确,那么你知道java和javac两个命令有什么用吗? 在了解这两个命令的作用 ...

  6. java获取当月有几天_你真的能在JAVA开发这条路上面一直坚持下去吗?

    JAVA为什么有前途? 过去的十多年,JAVA基本每年都是全世界使用人数第一的语言.全世界数百万的IT企业构建了庞大的JAVA生态圈,大量的软件基于JAVA开发. JAVA也被誉为"计算机界 ...

  7. java怎么运行_不要再问我Java程序是怎么执行的了!

    什么是Java虚拟机? 要弄明白Java程序的执行过程首先要了解一下Java虚拟机. 虚拟机是一种抽象化的计算机,通过在实际的计算机上仿真模拟各种计算机功能来实现的.Java虚拟机有自己完善的硬体架构 ...

  8. java比较炫的小程序_有人不建议学Java,有人不建议学Python,该怎么选择

    首先,编程语言都有自己所特定的应用场景,不同的编程语言有不同的优势,同样也会有相应的缺点,所以程序员往往需要掌握多门编程语言,以应对不同的开发场景.虽然Java和Python都属于全场景编程语言,但是 ...

  9. java洗衣店管理课程设计报告_课内资源 - 基于Java的洗衣店管理系统

    摘 要 随着科技的快速发展,人们的需求也是越来越多,为了方便对信息的管理我们小组就设计了一个洗衣店订单的管理系统. 洗衣店管理系统是典型的的信息管理系统,创建了六个类,分别是:Test类:Person ...

最新文章

  1. Webhooks上的一个简单方法:恐吓现在停止
  2. css左固定右自适应常用方法
  3. .net c# 正则表达式 平衡组/递归匹配
  4. shell换行合并多个文件_如何合并多个pdf文件?这里有合并PDF最简单的方法
  5. matlab fname pname,求大神帮我解释一下matlab最后几行是什么意思
  6. Java中去除字符串中空格的方法
  7. java服务 重启_Shell脚本自动重启Java服务
  8. 文言文亦能编程,该项目在GitHub摘得5500星!
  9. LACP链路聚合控制协议
  10. android 1s,android 与 小米1S刷机学习
  11. JSPSSM316的网上水果生鲜销售商城系统java毕业设计成品源码介绍
  12. Hack with Lenna | PIX与诗歌、自动驾驶和图像识别
  13. 超级记忆/图像数字记忆 110位数字图像转换表 81-90
  14. JS实现弹窗效果,超简单哦
  15. python如何执行部分代码_python如何运行代码
  16. 为什么买入不了创业版_为什么说不能买创业板?
  17. 黑群晖vmm专业版_群晖的ABB应用不错,但必须吐槽群晖 VMM功能,让你看看就是不让你好好用!!!...
  18. 基于matlab的锁相环频率合成器,基于MATLAB锁相环.ppt
  19. python学习笔记之序列,内含列表和元组的常用方法
  20. TcaplusDB君 · 行业新闻汇编(11月15号)

热门文章

  1. 三星S9勃艮第红如何为女人打造“第四维”魅力?
  2. 一个物体的固有频率有几个?
  3. 30岁还一事无成,你该怎么办?
  4. 韩国2018年GDP增速为2.7% 人均GNI或超3.1万美元
  5. 立方数 唯一分解 开方四舍五入
  6. 宏观结构分析之语篇模式视角
  7. python人工智能项目开发实例-人工智能学习路线图,整理近200个实战案例与项目...
  8. js计算数组中的最大值
  9. MARKETS AND MARKET LOGIC——The Market‘s Principles (5)
  10. 74hc165并行口转串口芯片学习