- VisualStudio2022插件的安装及使用-编程手把手系列文章
- pprof-在现网场景怎么用
- C#实现的下拉多选框,下拉多选树,多级节点
- 【学习笔记】基础数据结构:猫树
大纲 。
1.G1的YGC过程 。
2.YGC并行处理阶段的过程 。
3.YGC串行处理阶段的过程(一) 。
4.YGC串行处理阶段的过程(二) 。
5.整个YGC的执行流程总结 。
。
1.G1的YGC过程 。
(1)YGC相关的一些参数 。
(2)YGC和MixedGC、FGC之间的关系 。
(3)YGC使用的算法 + 新生代的垃圾回收流程 。
。
(1)YGC相关的一些参数 。
一.-XX:+UseG1GC 。
设置使用G1垃圾回收器.
。
二.-XX:G1HeapRegionSize 。
设置Region分区大小,最小值1M,最大值32M,且只能是2的n次幂.
。
三.-Xms和-Xmx或者InitialHeapSize和MaxHeapSize 。
设置堆内存最大值最小值.
。
四.-XX:NewSize和-XX:MaxNewSize 。
设置新生代最小值和最大值。注意,在G1里这个最大值最小值其实是可以不设置的。G1会自动计算出一个值:从5%的Region数量开始,慢慢增加到最大为60%的Region数量。一般不用指定新生代的最大值和最小值,按照默认的5%~60%即可.
。
五.新生代Region数量 。
下限-XX:G1NewSizeRercent,默认5% 。
上限-XX:G1MaxNewSizePercent,默认60% 。
。
六.新生代Eden和Survivor的比例:-XX:SurvivorRatio=n 。
默认为8,即eden : s1 : s2 = 8 : 1 : 1。这个比例和ParNew的原理是一致的,比如总的新生代是100个Region,那么Eden区有80个,两个S区各有10个.
。
七.-XX:MaxGCPauseMills=n 。
设置最大GC暂停时间,这是一个大概值,JVM会尽可能的满足此值。例如设置200ms,那么G1就会在每次GC时努力保证GC的停顿时间在这个范围内.
。
八.-XX:NewRatio=n 。
新生代与老年代的大小比例,默认值2。只设置一个NewRatio,与只设置一个Xmn是相当的。最好不要设置一个Xmn,或者最好不要单独设置一个NewRatio。因为这样会固定新生代大小,不利于停顿时间预测.
。
九.-XX:ParallelGCThreads=n 。
参与回收的线程数量,默认和CPU核数相等.
。
(2)YGC和Mixed GC、FGC之间的关系 。
G1中的YGC、Mixed GC以及FGC和ParNew + CMS是有相似之处的。比如,在新生代GC后,会有存活对象进入老年代。如果老年代对象占用达到了某个阈值,就会触发老年代的回收。在ParNew + CMS中,是直接触发FGC,而在G1中是触发Mixed GC.
。
在G1中首先会进行YGC,YGC会选择所有新生代的分区进行回收。当程序不断运行,存活对象越来越多,老年代的对象越来越多时,就会在某次YGC的时候,触发一个并发标记过程.
。
然后等待YGC和这次并发标记过程结束后,就会正式进入Mixed GC。Mixed GC会从老年代中选择部分回收价值比较高的Region进行回收,从而满足用户设置的MaxGCPauseMills值,当然Mixed GC也会回收所有新生代分区。当Mixed GC后,对象还是无法分配成功时,就会触发FGC。FGC会暂停程序运行,对整个堆进行全面的垃圾回收。FGC的回收会包括新生代、老年代、大对象等.
。
YGC、Mixed GC、FGC间的过程转换关系如下:
(3)YGC使用的算法 + 新生代的垃圾回收流程 。
一.不均匀的分区分布 。
二.对象在Eden区的分布 。
三.Eden区占满时触发YGC 。
四.标记存活对象 。
五.复制存活对象到Survivor区 。
六.回收垃圾对象 。
七.动态调整新生代区域Region数量 。
八.是否需要开启并发标记 。
九.新生代的垃圾回收流程结束 。
。
YGC使用的算法是复制算法,也就是会把新生代的所有Region按照Eden、Survivor做类型标识。在执行垃圾回收时,首先对存活对象进行标记。然后把存活对象复制到S区,接着就把所有的垃圾对象全部回收掉。注意:G1的Region分布,对于一个分代而言,不一定是连续的.
。
一.不均匀的分区分布 。
二.对象在Eden区的分布 。
三.Eden区占满时触发YGC 。
四.标记存活对象 。
首先从GC Roots出发,标记直接引用的对象,然后再一个一个标记GC Roots间接引用的对象.
五.复制存活对象到Survivor区 。
六.回收垃圾对象 。
如果是ParNew + CMS的新生代回收,其实到这里就基本上是结束了.
七.动态调整新生代区域Region数量 。
YGC回收掉的那些Region,是有可能会成为自由分区的。因为YGC回收后会动态判断,如果需要更多的Region就进行增加。如果回收时间太长,发现YGC下一次可能没有那么强的能力,那么G1就会减少几个Region.
。
八.是否需要开启并发标记 。
如果老年代的使用率达到了阈值,就开启并发标记.
。
九.新生代的垃圾回收流程结束 。
以上就是G1新生代垃圾回收的基本流程.
。
(4)总结 。
G1的YGC过程:
一.YGC相关的一些参数 。
二.YGC和MixedGC、FGC是什么关系 。
三.YGC使用的算法 + 新生代的垃圾回收流程 。
1.不均匀的分区分布 。
2.对象在Eden区的分布 。
3.Eden区占满时触发YGC 。
4.标记存活对象 。
5.复制存活对象到Survivor区 。
6.回收垃圾对象 。
7.动态调整新生代区域Region数量 。
8.是否需要开启并发标记 。
9.新生代的垃圾回收流程结束 。
。
2.YGC并行处理阶段的过程 。
(1)YGC的并行处理是什么 。
(2)GC Roots并行标记及RSet并行更新 。
(3)YGC并行标记阶段不仅会标记还会复制 。
(4)完成初始标记后的处理——将直接引用的对象的字段入栈 。
(5)遍历栈中的字段找出存活对象并复制到S区然后清理对象 + 清空栈 。
。
YGC有并行处理的过程以及串行处理的过程.
。
(1)YGC的并行处理是什么 。
YGC的过程,肯定是要做一些并行化处理的,否则速度就会比较慢。比如,标记对象时就不能一个一个对象去查找标记。所以,会有线程对GC Roots直接引用的对象进行标记.
。
(2)GC Roots并行标记及RSet并行更新 。
不仅在对GC Roots引用的对象进行标记时,会使用并行处理的方式。在对RSet进行更新时,也会用并行处理的方式.
。
G1在进行YGC时:会从RSet和GC Roots出发遍历所有新生代对象,然后标记存活对象。由于RSet的更新不一定会在YGC前就更新完毕,所以在YGC并行处理这个阶段,还要对RSet做并行处理的更新。即把DCQS里还没处理完毕的跨代引用关系变更,更新到RSet里面.
更新RSet完成后,再从RSet出发,去标记被RSet指向的老年代空间里的对象直接引用的新生代对象.
整个标记GC Roots + 更新RSet的过程,是由多个线程一起并行处理的。比如现在有4个GC线程参与垃圾回收,那么就会有两个线程从GC Roots出发去标记对象,有两个线程去消费DCQS然后更新RSet。接着Rset更新完毕后,就把RSet作为GC Roots继续去执行对象标记工作.
。
(3)YGC并行标记阶段不仅会标记还会复制 。
在YGC的并行标记阶段,不仅仅会根据GC Roots + RSet来追踪所有直接引用的对象。由于在执行YGC的过程中,复制操作和标记操作是同时进行的。所以在用GC Roots标记直接引用的存活对象时,也会进行复制操作。比如发现4个对象是由GC Roots直接引用的.
此时通过GC Roots找到这4个对象后,就会复制它们到一个Survivor区.
所以YGC里的复制算法,并不是等待全部标记完成,再去复制对象。而是找到一个直接引用的存活对象,就会复制到Survivor里了.
。
另外,把RSet作为GC Roots的意思是:RSet中映射到卡表对应的卡页中的所有对象都会作为GC Roots。因为卡页本身很小,对象数量也很少,所以可以把RSet都作为GC Roots。然后找到这些GC Roots直接引用的对象,再复制到Survivor区.
。
(4)完成初始标记后的处理——将直接引用的对象的字段入栈 。
仅仅处理这些GC Roots直接引用的对象还是不够的,因为还有很多对象会被它们间接引用。间接引用的对象,也需要全部找到并进行标记.
。
那么在并行处理阶段,GC线程还需要做的另外一件事就是:把刚刚找到的被GC Roots直接引用的哪些对象的字段Field,全部都给放入一个栈里面.
。
为什么要这么做?因为要把这些对象引用的所有对象都找到才行,找到它们引用的对象才能找到所有存活的对象。所以在把GC Roots + RSet直接引用的对象复制到S区时,就会把它们的所有字段放入一个栈中.
(5)遍历栈中的字段找出存活对象并复制到S区然后清理对象 + 清空栈 。
等到所有的GC Roots + RSet直接引用的对象都复制完毕后,再逐一对栈中的字段Feild进行遍历,找到所有存活的对象,然后再把找到的存活对象放入Survivor区中.
最后一口气回收掉所有的垃圾对象.
此时栈就会被清空掉了.
至此,YGC的并行操作基本已经结束,基本上YGC其实已经结束了。但是,实际上还会有后续的很多操作。比如以下操作就是在YGC的串行执行过程中需要做的,需要更新RSet、RSet卡表、释放被回收垃圾占用的Region、动态调整新生代分区数量来实现停顿预测模型等.
。
(6)总结 。
YGC并行处理阶段的过程:
一.YGC的并行处理是什么 。
二.GC Roots并行标记及RSet并行更新 。
三.YGC并行标记阶段不仅会标记还会复制 。
四.完成初始标记后的处理——将直接引用的对象的字段入栈 。
。
YGC的并行处理阶段具体会做的事情:
一.并行更新RSet 。
二.将更新完RSet加入GC Roots进行并行标记 。
三.并行复制直接引用的存活对象进入S区 。
四.将直接引用的存活对象的所有字段入栈 。
五.遍历栈的所有字段寻找所有存活对象 。
六.复制所有存活对象进入S区 。
七.清空全部垃圾对象 。
八.清空栈里的所有字段 。
。
3.YGC串行处理阶段的过程(一) 。
(1)YGC中的串行处理是什么 。
(2)YGC中的串行处理操作有哪些 。
一.软引用、弱引用、虚引用的处理 。
二.整理卡表 。
三.Redirty操作——清理旧RSet建立新RSet 。
四.释放分区 。
。
(1)YGC中的串行处理是什么 。
所谓串行处理,就是要一步步操作,否则就可能会出现错乱的一些操作。JVM会对垃圾回收中的一些操作使用串行化的处理方式。可能因为这些操作会有前后影响、或消耗的时间很少,所以才用串行化。当然不排除JVM后面可能会把这些操作优化成并行化的处理方式.
。
其中G1中的GC Roots追踪、RSet更新,这两个操作是可以并行进行的,因为这两个操作基本上不会出现互相影响的情况。但是YGC的其他一些操作,是有可能会出现先后影响的.
。
(2)YGC中的串行处理操作有哪些 。
一.软引用、弱引用、虚引用的处理 。
该操作是把这些引用中使用的存活对象也复制到新分区,否则就会出错。YGC中的并发处理阶段针对的是强引用对象.
。
软引用的回收时机:在第一次FGC时,是不回收软引用的。只有在第二次Full GC时,才会回收软引用.
。
在YGC执行串行处理操作时:就会把新生代里被这类引用给引用到的对象复制到Survivor区中.
二.整理卡表 。
卡表是一个全局卡表。在新生代在回收后,有些对象已被回收清除了,有些对象已经换了位置。这时就要把卡表中这些对象的描述数据也给清除掉和更新掉.
。
因为卡表中是一个字节描述512字节的内存空间。如果某内存空间被清除了,那么卡表的描述数据也需清空,否则会出错.
。
所以整理卡表的操作就是把已清理过的Region对应的卡表进行清空,同时把对象复制后所在的Region对应的卡表也进行修改,从而保证卡表中的描述数据是正确的。整理卡表的这个过程是很快的.
三.Redirty操作——清理旧RSet建立新RSet 。
这一步的主要操作,其实就是重构一下Rset。在做完垃圾回收后,新生代对象因为复制,其地址已经发生变化了。那么老年代引用的新生代对象所在Region的RSet此时还没有修改,因此需要把这个旧的RSet进行清理,然后建立一个新生代对象所在的新Region的RSet。重构RSet的过程也是很快的.
。
四.释放分区 。
新生代GC,需要把所有的非Survivor区的新生代Region都给清理掉。此时它们还被标记为Eden,或者Survivor(原本就可能有一些垃圾对象)。清理后,需要把这些Region分区给释放掉,否则需要分区时可能不够用。释放分区就是清空这些分区的标记,然后把清空后的分区加入到自由分区列表.
。
(3)总结 。
YGC串行处理阶段的过程:
一.YGC中的串行处理是什么 。
二.YGC中的串行处理操作有哪些 。
1.软引用、弱引用、虚引用的处理 。
2.整理卡表 。
3.Redirty操作——清理旧RSet建立新RSet 。
4.释放分区 。
。
(4)问题 。
如果我们是G1的开发者,在上面的流程结束之后还需要做什么?G1本身的设计思路就是,要垃圾回收优先,要满足系统的停顿时间。那么在GC之后最重要的事情是什么呢?
。
一.停顿预测模型和Region数量分配有关; 。
二.RSet处理时是比较耗时的,GC开启时Refine线程就会暂停,由GC线程来继续执行后续的操作。那么对于这个RSet、DCQ、DCQS的处理,是否需要调整?
三.是否需要扩展内存?
。
4.YGC串行处理阶段的过程(二) 。
(1)尝试对大对象进行回收(性价比很高) 。
(2)尝试扩展内存 。
(3)调整新生代分区的数目及Refine线程阈值 。
(4)尝试启动并发标记 。
。
(1)尝试对大对象进行回收(性价比很高) 。
一.为什么要对大对象尝试进行回收操作 。
原因一:大对象本身占用很多空间的,最少也会占1/2的Region 。
假如大对象能够回收,就顺带把它回收掉,这样就能腾出一块非常可观的空间出来了.
。
原因二:大对象回收起来不麻烦 。
因为大对象创建时是单独存储在一个分区(多个分区)的,属于单独存储.
。
二.如何判断大对象是否存活 。
由于每个Region都维护了一个RSet,并且RSet里存储的是引用关系信息。那么在YGC的串行处理阶段查看大对象所在Region的RSet,就能知道是否有其他对象在引用了.
。
注意大对象所在的Region的RSet不会有很多内容,最多就是两个对象被引用的关系.
。
所以如果大对象没有横跨多个分区,则只需判断一下大对象所在的Region的RSet里是否有内容,就可以判断大对象是否存活了.
。
如果大对象横跨多个分区,那么直接判断大对象所在的第一个Region的RSet里是否有内容,就可以知道大对象是否被引用了.
。
通过简单的判断就可能回收大量的空间,性价比非常高。所以要在YGC的串行处理阶段尝试一下对大对象进行回收.
。
(2)尝试扩展内存 。
前面介绍新生代内存时,介绍过可能会对新生代内存进行扩展,在YGC的串行处理阶段尝试扩展内存就是扩展新生代内存的时机之一.
。
完成YGC后会统计一下执行这次YGC的花费时间,而且还会统计一下在执行YGC前的系统运行总时间。于是就可以判断,这次YGC执行时间和系统运行总时间的比例是否合理。如果不合理,就要考虑扩展一下新生代内存,如果合理就没必要扩展了.
。
对应的参数是:GCTimeRatio和G1ExpandByPercentOfAvailable。其中GCTimeRatio是指:程序运行时间与YGC时间的比例。如果YGC时间占程序运行时间比例超过10%,就说明要扩展新生代内存.
。
为什么YGC时间占程序运行时间的比例超过10%就要扩展新生代内存?因为YGC时间占比超过10%,就说明要么YGC频繁、要么YGC时间太长。如果新生代空间足够大,加上G1会自己动态调整新生代分区的数量,那么就是YGC太频繁导致YGC的时间占程序运行时间比例超过10%.
。
YGC过于频繁,必然会导致判断出大量对象存活,相当于变相拖慢YGC。YGC中真正耗时的不是清理大量垃圾对象的过程,而是进行标记的过程。YGC中存活对象越多,进行标记的过程就越长。YGC越频繁 -> 说明新生代很快满了 -> 说明新生代需要扩展内存 。
。
如果YGC时间占程序运行时间的比例没有超过10%,则暂时不需要扩展,扩展的内存大小和G1ExpandByPercentOfAvailable有关.
。
(3)调整新生代分区的数目及Refine线程阈值 。
一.调整新生代分区数目 。
这个是YGC串行处理阶段的一个重点,因为对于G1来说,控制停顿时间是非常重要的.
。
要想控制好停顿时间:只能在系统运行时间和YGC过程中各个步骤的耗时上进行综合考量。综合考量后还要进行动态调整,这样才能保证停顿时间是可以被满足的.
。
那么在YGC后,首先就需要判断一下:现在的YGC耗时、YGC能力能否让下一次GC满足预期停顿时间。如果不能的话,那么就需要把新生代分区减少一些,不然就满足不了了。如果远远没达到停顿预测时间的阈值,那么就可以增加一些新生代分区.
。
所以这一步,就会根据当前YGC的执行时间和目标停顿时间,进行预测。看下一次YGC最多能回收多少分区,然后和当前新生代的总分区数对比。如果下次最多能回收的分区和当前新生代总分区数差不多,则无需调整。如果发现预测出来下一次YGC能回收1000个分区,而现在才600个分区。那么就可以多增加几个分区到新生代里,避免浪费堆内存.
。
二.调整RefinementZone的阈值 。
关于DCQS、DCQ和Refine线程的处理:如果DCQ比较多,则需要启动多个Refine线程去进行处理。而且在YGC开始时,这些Refine线程就会暂停,并且由YGC线程接管其工作来处理后续的DCQ.
。
如果YGC线程处理DCQ的时间过多,那么代表了什么?代表Refine线程的数量,或者DCQS的四个区域设置得不合理。如果设置合理,Refine线程在对应的区域中,就可满足DCQ消息的处理。此时YGC线程最多就是进行少部分的收尾工作,但现在YGC线程还需要大量的时间去处理DCQ消息,那么就说明:要么这几个DCQS的阈值设置得过大了、要么Refine线程太少了.
。
Refine线程理论上是不能在GC过程中动态调整上限的,所以我们只能调整DCQS的白绿黄红几个阈值的大小,通过白绿黄红来匹配Refine线程的处理能力.
。
比如把DCQS的各个阈值给降下来,然后把总长度也降下来。让系统线程也帮忙处理DCQ,这样就可以让GC线程的压力小一点。所以如果YGC处理DCQ时间过长,会导致DCQS的长度和阈值动态减小.
。
(4)尝试启动并发标记 。
这个过程也是一个尝试的过程。因为新生代的回收是一直在进行的,老年代的对象也是一直在累积的.
。
如果老年代对象累积到一定程度,那么此时就需要回收一部分老年代的垃圾对象,否则内存使用率就会太高。所以,在老年代达到45%的内存使用率时,一次YGC结束后就会开启一个并发标记过程.
。
如下图示:老年代占用内存达到阈值的判断,就是判断是否要进入MGC + YGC过程。如果成功启动了并发标记,就意味着接下来要进入Mixed GC了.
。
5.整个YGC的执行流程总结 。
YGC算法的流转过程:
。
最后此篇关于G1原理—4.G1垃圾回收的过程之YoungGC的文章就讲到这里了,如果你想了解更多关于G1原理—4.G1垃圾回收的过程之YoungGC的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我是 C 新手,还没有真正掌握 C 何时决定释放对象以及何时决定保留对象。 heap_t 是指向结构堆的指针。 heap_t create_heap(){ heap_t h_t = (heap
我有一个问题,我不知道如何解决。问题是: char * ary = new Char[]; ifstream fle; fle.open(1.txt, ios_base::binary); fle.s
假设我在 C# 中有字符串:“我看不到你……” 我想删除(替换为空等)这些“’”符号。 我该怎么做? 最佳答案 那个“垃圾”看起来很像有人将 UTF-8 数据解释为 ISO 8859-1 或 Wi
我无法在解析方法中更改蜘蛛设置。但这绝对是一种方式。 例如: class SomeSpider(BaseSpider): name = 'mySpider' allowed_domains
在开始之前,我们先回顾一下堆是个什么玩意,大家可能都知道,我们每天创建的Java对象几乎都存放在堆上面,所以说堆是一个巨大的对象池一点都不过分,在这个对象池里面管理者数据巨大的对象实例。 在对
我想知道为什么 printf() 在提供数组且没有格式化选项时成功打印字符数组,但在使用整数数组时编译器会抛出警告并打印垃圾值。 这是我的代码: #include int main() { c
我正在研究 Scrapy 库并尝试制作一个小爬虫。 这是爬虫的规则: rules = ( Rule(LinkExtractor(restrict_xpaths='//div[@class="w
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: Printing a string to a temporary stream object in C++
这个问题在这里已经有了答案: Are WebGL objects garbage collected? (2 个答案) 关闭 3 年前。 在 WebGL 中,纹理的创建和销毁使用: WebGLTex
我继承了以下代码: (为保护无辜者更改了一些名称。) package foo.bar.baz; import javax.swing.JPanel; //Main panel in the GUI c
如果我没记错的话,在某些情况下,Java 中的 lambda 会生成为匿名类实例。例如,在这段代码中,lambda 需要从外部捕获一个变量: final int local = 123456; lis
我正在阅读托管代码中的内存泄漏,想知道是否可以在 C# 不安全代码中创建它? unsafe { while(true) new int; } 我不确定如果它作为不安全代码运行,是否会被 GC
假设我有以下用 HTML 编写的网页(仅正文部分): ... function fn() { // do stu
我想知道是否有简单的命令可以删除在 latex 编译过程中生成的所有不必要的文件,例如.aux、.log 等 最好将它链接到常规的 Latex 构建命令,这样在我点击“编译”后,垃圾文件就会被删除。
Java 在 Java7 中引入了带有字符串的 switch case。我想知道使用这样的开关盒是否会产生垃圾。 例如在我的程序中, String s = getString(); switch(s)
Cevelop将 char junk 作为“未初始化的变量”对象。在这种情况下,解决问题的正确方法是什么? friend std::ostream& operator>(std::istream&
关闭。这个问题需要debugging details .它目前不接受答案。 编辑问题以包含 desired behavior, a specific problem or error, and t
我正在编写一个发送和接收纯文本的小型 boost asio tcp 服务器和客户端。通信或多或少是请求响应。在测试期间,我想我只是向服务器发送垃圾数据,向它发送 100.000 个请求。 客户端发
我正在使用 SAX 来读取/解析 XML 文档,并且它工作正常,除了这个特定的站点,在该站点中 eclipse 告诉我“文档元素之后的垃圾”并且我没有返回任何数据 http://www.zachblu
这是我的 Scrapy 爬虫代码。我正在尝试从网站中提取元数据值。没有元数据在一个页面上出现多次。 class MySpider(BaseSpider): name = "courses"
我是一名优秀的程序员,十分优秀!