GC.Collect如何影响垃圾回收

简介:

根据垃圾回收的算法,对象在内存中是按代的方式存放的,通常情况下,当第0代沾满分配的空间的时候(比如是256k),GC就会启动去回收第0代对象,幸存的第0代对象会被放入第1代中去,第1代的对象要等到放满了才会收集,因此,越是年轻的代越是被频繁的收集,由于通常情况下GC只收集第0代对象,既保证了可回收较多的内存,又忽略了老一代的对象,从而加快了垃圾回收的速度,提升了性能。

因此当调用gc.collect的时候,相当于强制的对所有代,不管年轻还是老的都执行一次回收。由于垃圾回收器在回收的资源的时候,正在执行托管代码的线程都会被挂起,具体的细节相当复杂,因为有的线程运行在不安全的点,CLR不能执行垃圾回收,因此CLR会采用线程劫持技术,即通过修改线程栈的方法,来做垃圾回收。这种复杂性使得性能降低。除非确定大量的旧对象死亡,才考虑调用这个方法。

所以,在一般情况下,尽量不要干预垃圾回收器工作,即尽量避免主动调用GC.Collect。

由于垃圾回收是异步的,CLR有一个专用的线程负责垃圾回收,因此,即使调用GC.Collect,也并不是实时的调用了Finalize,因此要保证确实调用了析构方法,可以使用语句GC.WaitForPendingFinalizers()。

下面是一段代码,通过注释掉

GC.Collect();

GC.WaitForPendingFinalizers();

语句,看出端倪。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
using  System;
using  System.Collections.Generic;
using  System.Linq;
using  System.Text;
using  System.Linq.Expressions;
using  System.Reflection;
namespace  ConsoleApplication1
{
     class  Program
     {
         static  void  Main( string [] args)
         {
             AA aa1 =  new  AA( "1" );
             AA aa2 =  new  AA( "2" );
             AA aa3 =  new  AA( "3" );
             aa1 =  null ;
             aa2 =  null ;
             //GC.Collect();
             //GC.WaitForPendingFinalizers();
             var  tmp = aa3;
         }
     }
     public  class  AA
     {
         public  string  id =  "" ;
         public  AA( string  s)
         {
             id = s;
             Console.WriteLine( "对象AA_"  + s +  "被创建了" );
         }
         ~AA()
         {
             Console.WriteLine(id +  " 析构函数被执行了" );
         }
     }
}

当语句被注释掉的时候,虽然aa1和aa2都设成了null,但是垃圾回收并不是马上就把它们回收掉。对象可能都被放在第0代上,等进程结束的时候,由垃圾回收器一起回收。所以输出如下,顺序是321

e1373249592f420e854f3b035ddf3fd7

但是当取消注释后,由于强制垃圾回收时,aa1对象和aa2对象都是null,因此就把它们回收掉了。顺序就是213了。

b0b99d8b1eec4114b26e7fa195225f4b

注意,如果aa1和aa2不设成null,那么强制回收时,并不认为这2个对象可以回收。因此还是会等到进程结束的时候才会回收。














本文转自cnn23711151CTO博客,原文链接:http://blog.51cto.com/cnn237111/1343004 ,如需转载请自行联系原作者


相关文章
|
10月前
|
Java 关系型数据库 数据库连接
简单易懂的 MyBatis 分库分表方案
本文介绍了一种基于 MyBatis 框架的数据库分库分表方案——shardino。不同于复杂插件方式,该方案通过客户端代码包装实现简便易懂的操作,显式处理分库分表逻辑,确保开发者清晰了解数据分布。项目地址:[https://github.com/pyloque/shardino](https://github.com/pyloque/shardino)。方案中,帖子表按 userId 字段 hash 分为 64 张表,平均分配到多个主从库中,配置文件管理 MySQL 组对象,支持读写分离和权重随机选择从库。代码示例展示了如何计算 partition number 并进行具体操作。
326 22
简单易懂的 MyBatis 分库分表方案
|
XML Java 数据格式
Spring IOC—基于XML配置Bean的更多内容和细节(通俗易懂)
Spring 第二节内容补充 关于Bean配置的更多内容和细节 万字详解!
714 18
Spring IOC—基于XML配置Bean的更多内容和细节(通俗易懂)
|
JavaScript 前端开发 UED
(详解错误情况,及解决方法)Vue 数据更新了但页面没有更新的情况
(详解错误情况,及解决方法)Vue 数据更新了但页面没有更新的情况
727 0
|
人工智能 vr&ar 计算机视觉
CVPR 2024:让图像扩散模型生成高质量360度场景,只需要一个语言模型
【6月更文挑战第20天】CVPR 2024研究表明,结合语言模型的图像扩散模型能高效生成360度全景图像,减少对标注数据的依赖。该框架利用语言模型的语义信息引导细节丰富的图像生成,解决了传统方法的标注难题。然而,方法的准确性和计算资源需求是挑战。这一进展推动了VR/AR图像生成技术的发展。[论文链接](https://arxiv.org/pdf/2406.01843)**
248 6
|
JavaScript API 定位技术
使用vue3结合高德实现定位功能
使用vue3结合高德地图实现定位功能
13913 59
使用vue3结合高德实现定位功能
|
并行计算 Linux 异构计算
分享一款刚开源上线3天的音乐人声分离工具!无需联网!页面化操作!
分享一款刚开源上线3天的音乐人声分离工具!无需联网!页面化操作!
442 1
|
存储 前端开发 API
基于SpringBoot+Vue的智能在线考试系统的设计与实现
基于SpringBoot+Vue的智能在线考试系统的设计与实现
635 0
|
机器学习/深度学习 人工智能 达摩院
阿里达摩院入职要求
阿里达摩院入职要求
|
专有云
阿里云产品体系分为6大分类——企业应用——分为11类——专有云
阿里云产品体系分为6大分类——企业应用——分为11类——专有云自制脑图
365 0
|
编解码 Java Android开发
Android11.0(R) MTK6771 平板横屏方案修改(强制app横屏 + 开机logo/动画+关机充电横屏 + RecoveryUI 横屏)
Android11.0(R) MTK6771 平板横屏方案修改(强制app横屏 + 开机logo/动画+关机充电横屏 + RecoveryUI 横屏)
743 0