zoukankan      html  css  js  c++  java
  • 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的大对象使用。

  • 相关阅读:
    个人作业——软件工程实践总结作业
    团队作业——Beta冲刺5
    Beta冲刺博客集合贴
    团队作业——Beta冲刺4
    团队作业——Beta冲刺3
    团队作业——Beta冲刺2
    团队作业——Beta冲刺1
    Alpha事后诸葛亮(阳光普照队)
    软件产品案例分析
    个人作业——软件工程实践总结作业
  • 原文地址:https://www.cnblogs.com/zwt-blog/p/5416607.html
Copyright © 2011-2022 走看看