page contents

浅谈Java的System.gc()实现

我们都知道System.gc()用于调用垃圾收集器。很久之前我一直认为执行System.gc()之后,虚拟机会立刻垃圾回收。 抱歉,我理解错了。 直到看完System.gc()的源码之后才搞清楚,执行System.gc()函...

我们都知道System.gc()用于调用垃圾收集器。很久之前我一直认为执行System.gc()之后,虚拟机会立刻垃圾回收。

抱歉,我理解错了。

直到看完System.gc()的源码之后才搞清楚,执行System.gc()函数的作用只是提醒或告诉虚拟机,希望进行一次垃圾回收。

至于什么时候进行回收还是取决于虚拟机,而且也不能保证一定进行回收(如果-XX:+DisableExplicitGC设置成true,则不会进行回收)。

然后在简单说一下什么样的对象会被gc()回收:sum的hotspot虚拟机是通过可达性分析算法来判断对象是否需要回收。这里不重点讲述可达性分析算法,之后在单独写一篇博客来介绍。

先来简单看一下System.gc()的源码吧:


浅谈Java的System.gc()实现

简单看一下gc()函数的注释(翻译的不对的地方望见谅,水平有限)

Runs the garbage collector.

Calling the <code>gc</code> method suggests that the Java Virtual

Machine expend effort toward recycling unused objects in order to

make the memory they currently occupy available for quick reuse.

When control returns from the method call, the Java Virtual

Machine has made a best effort to reclaim space from all discarded

objects.

运行垃圾收集器。

调用gc()函数表明Java虚拟机花费了很多精力来回收未使用的对象,以使它们当前占用的内存可用于快速重用。

当控制从方法调用返回时,Java虚拟机已尽最大努力从所有丢弃的对象中回收空间。

在注释中也明确的说明了,调用System.gc()等同于调用Runtime.getRuntime().gc()。

好吧,我们接着来分析Runtime.getRuntime().gc():


浅谈Java的System.gc()实现

好吧,原来gc()是native方法,在java层面只能看到这么多了,先来看一下注释的意思吧。

意思和System.gc()的注释差不多,需要注意的是,在注解中也说明了,需要在单独的线程自动执行。

正好我电脑上有OpenJDK8的代码,我们接着去Hotspot里看看System.gc()是怎么实现的吧

先下载OpenJDK的源码,我下载的是OpenJDK8的源码,在jdk/src/share/native/java/lang 目录中有一个 Runtime.c 文件


浅谈Java的System.gc()实现

没错就是这个函数(抱歉,我不能解释我是怎么知道就是这个函数的,因为我是根据名称推测的。。。),很简单,只是调用了JVM_GC这个函数。

那我们接着看JVM_GC这个函数实现的逻辑吧,其实我不用C和C++,也是咨询了JVM大神才定位到JVM_GC函数的定义的。

在hotspot/src/share/vm/prims 目录中有一个 jvm.cpp 文件


浅谈Java的System.gc()实现

其实我不是太能看懂,我只能带着我的疑问一步步来推测了,DisableExplicitGC 是在哪定义的呢?默认是true还是false呢?

来跟大家说一下怎么找到 DisableExplicitGC 这个函数的定义:在OpenJDK目录下执行 grep -nr "DisableExplicitGC" . 。唉,因为不会,所以只能用这么笨的办法了,见笑。

在hotspot/src/share/vm/runtime 目录中有一个 globals.hpp 文件,这里面就有 DisableExplicitGC 的定义


浅谈Java的System.gc()实现

好了,这个疑问解决了,默认设置成了false,所以在不修改 DisableExplicitGC的情况下,会执行 if 里面的代码的。接着分析 if 里面的代码啦。

调用了collect()函数,所以我们看一下这个函数的实现:


浅谈Java的System.gc()实现


浅谈Java的System.gc()实现

如果should_do_concurrent_full_gc返回true,那会执行collect_mostly_concurrent做并行的回收,should_do_concurrent_full_gc的实现也就不在多说了,代码贴出来了。有兴趣的朋友可以继续跟踪下去。

  • 发表于 2020-02-17 15:28
  • 阅读 ( 687 )
  • 分类:Java开发

你可能感兴趣的文章

相关问题

0 条评论

请先 登录 后评论
Pack
Pack

1135 篇文章

作家榜 »

  1. 轩辕小不懂 2403 文章
  2. 小柒 1474 文章
  3. Pack 1135 文章
  4. Nen 576 文章
  5. 王昭君 209 文章
  6. 文双 71 文章
  7. 小威 64 文章
  8. Cara 36 文章