无论通过看书或是听别人讲关于垃圾回收的事情,我们或多或少都有一些了解,但是我们了解的内容是正确的吗,今天我要做的事情就是这样,证明垃圾回收规则。
垃圾回收原理
简单说一下垃圾回收的基本原理:
1:Clr将heap分成3个代龄区域: Gen 0、Gen 1、Gen 2。
2: 采用Mark Sweep算法进行回收。
3:GC触发条件:
- CLR检查到第0代超出预算时会触发一次GC;
- 代码显式调用System.GC的静态Collect方法;
- Windows报告低内存情况;
- CLR正在卸装AppDomain;
- CLR正在关闭;
代码
代码通过Collect方法主动回收对象:
1:第一步创建两个对象,zhangsn和lisi。并且这两个对象进入0代堆。
2:第二步回收zhangsn并且将lisi移动到1代堆。
3:第三步将lisi移动到2代堆。
验证-对象进入0代堆
1:运行程序,创建两个对象,zhangsn和lisi,如下图所示:
2:通过windbg工具(命令:!clrstack -l)查看两个对象在内存中的地址,如下所示:
3:通过windbg工具(命令:!eeheap -gc)查看内存堆情况,如下所示:
4:通过地址比较,我们可以看到zhangsan和lisi都进入0代堆,因为zhangsan和lisi的地址都大于0代堆:
下图是0代堆,1代堆,2代堆在内存中的位置,zhangsan和lisi都落入到了0代堆中,如下所示:
验证-zhangsan被回收,lisi进入1代堆
1:运行程序,回收zhangsn并且将lisi移动到1代堆,如下图所示:
2:通过windbg工具(命令:!clrstack -l)查看两个对象在内存中的地址,如下所示:
3:通过windbg工具(命令:!eeheap -gc)查看内存堆情况,如下所示:
4:通过地址比较,我们可以看到zhangsan已经被回收,lisi都进入1代堆,因lisi的地址大于1代堆小于0代堆:
下图是0代堆,1代堆,2代堆在内存中的位置,lisi落入到了1代堆中,如下所示:
验证-lisi进入2代堆
1:运行程序,lisi移动到2代堆,如下图所示:
2:通过windbg工具(命令:!clrstack -l)查看两个对象在内存中的地址,如下所示:
3:通过windbg工具(命令:!eeheap -gc)查看内存堆情况,如下所示:
4:通过地址比较,我们可以看到lisi都进入2代堆,因lisi的地址大于2代堆小于1代堆:
下图是0代堆,1代堆,2代堆在内存中的位置,lisi落入到了2代堆中,如下所示:
结论
因为这次实验采用的是Collect主动回收所有代的对象,所以得出如下结论:
1:新对象(非大对象)会被放入0代堆中(zhagnsan,lisi)。
2:发生垃圾回收时,对象没有被root对象引用,则会被回收掉(zhangsan)。
3:发生垃圾回收时,被root引用的对象,则会移动到下一代堆中(lisi)。
哪里说的不对,还请请教!