JVM-为什么YoungGen使用的堆在gc之间减少了?

时间:2018-12-13 08:12:47

标签: java apache-spark memory-management garbage-collection jvm

下面是一些Apache Spark执行器的GCViewer图:

  1. 老一代用过的堆
  2. 年轻一代用堆
  3. GC时间
  4. 所讨论的现象

我尝试了解(4)中的斜率。为什么要在使用整个年轻代堆之前先启动gc(如先前的gcs阶段)?为何它会在逐渐增加之前先保持约5分钟单调减少?我认为,例如,如果分配了非常大的对象(例如,从io套接字读取),则会发生这种情况。但这可能是错误的,因为那之后老一代并没有改变。我并不特别在乎这个示例,而只是在学习更多有关jvm内存管理的知识。

enter image description here

1 个答案:

答案 0 :(得分:1)

我能想到的一种可能的解释是线程本地分配块(TLAB)的结果。为了避免争用多个线程使用的指向Eden空间的单个指针,每个应用程序线程都有自己的指向Eden空间中的内存块的指针。当TLAB用完时,将分配一个新的TLAB(这也可能涉及分配一个更大的块,以便可以消除线程分配率之间的差异)。

这样做的含义是,当需要发生GC时,TLAB中通常会存在未使用的空间(因为一个线程需要一个新的TLAB,并且没有足够的空间供其使用)。您可能会遇到由于其他线程的分配率而导致TLAB中剩余空间增加的情况。

由于这是年轻一代的全部,因此幸存者空间中某些物体的保有权使用率可能会导致像这样的空间使用量减少。

不幸的是,从提供的数据来看,不可能给出确切的答案。