C#中考虑为大对象使用弱引用

  1.无论怎样尽力,我们总是会使用到某些需要大量内存的数据,而这些内存并不需要经常访问。或许你需要从一个大文件中查找某个特定的值,或者算法需要一个较大的查询表。这时,你也许会采用2中不太好做法:第一种是创建一个本地变量,然后在每次执行该算法时都生成一大块垃圾;第二种则是创建一个成员变量,在很长一段时间内都占用着这一大块内存。很多时候这两种做法都不是非常好。

  2.有没有一种好的选择来处这个问题呢?答案是创建弱引用,弱引用的对象和垃圾对象差不多,程序会告诉垃圾收集器该对象可以被回收,不过在回收之前你仍旧有一个引用,可以用来在需要的时候访问到该对象。若能合理使用该策略,弱引用将和垃圾收集器协同工作,从而优化内存使用。如下代码说明:

    /// <summary>
    /// 大对象类型
    /// </summary>
    public class MyLargeClass
    {
        private int[,] matrix;
        private ushort matrixXDimension;
        private ushort matrixYDimension;
        public string reallyLongMessage;

        private static WeakReference _weakObj;
        public static object WeakObj
        {
            get
            {
                return _weakObj?.Target;//“?.”语法为C#6.0的新特性,对应.Net Framework 4.6。表示若为空则返回空,若不为空则获取其相应的属性值
            }
            set
            {
                _weakObj = new WeakReference(value);//弱引用的创建
            }
        }

        public MyLargeClass(ushort matrixXDimension, ushort matrixYDimension)
        {
            this.matrixXDimension = matrixXDimension;
            this.matrixYDimension = matrixYDimension;
            matrix = new int[matrixXDimension, matrixYDimension];
        }

        private void Initialize()
        {
            //TODO 做更多的数据初始化
            //对matrix二维数组的数据加载
        }

        public long Calculate()
        {
            //TODO 处理对matrix对象计算结果
            return matrixXDimension * matrixYDimension * 1000;
        }

        public static void Execute()
        {
            //从弱引用中获得大对象,若没有弱引用或弱引用已被回收,此时就需要重新创建对象。
            MyLargeClass myLargeObject = WeakObj as MyLargeClass;//myLargeObject可为本地变量,也可为成员变量
            if (myLargeObject == null)
            {
                myLargeObject = new MyLargeClass(1000, 1000);
            }
            //TODO 对myLargeObject对象中的数据进行处理

            //处理完成后,把大对象归到弱引用,然后把myLargeObject设置为null,表明大对象无引用。此时,系统会认为myLargeObject所指向的对象可以被垃圾收集。若垃圾收集器在此时运行,那么将会回收该对象。不过若是在回收之前,你再次需要该对象,只需如上面的4行代码处理即可。
            WeakObj = myLargeObject;
            myLargeObject = null;
        }
}

  3.从上面的Execute()方法的2处代码中,可以看出对象的使用与清理交给了运行时处理,这是简单的使用场景,不过很多时候现实并不是那么简单,很少有对象会完全与外界隔离。任何一个大对象都包含了对其他对象的引用。若一个大对象被标记为弱引用,则其内部引用对象也将被标记为弱引用。被标记的弱引用对象就会被回收(不是马上回收)。

  4.对于实现了IDisposable的对象,则不应该使用弱引用,因为被释放的对象已经不能再弱引用,且在弱引用中你也不能调用Dispose()。所以弱引用应该配合非IDisposable的大对象使用。

原文地址:https://www.cnblogs.com/zwt-blog/p/5416607.html