在Winforms项目中垃圾收集过于激进

时间:2013-11-15 12:43:19

标签: c# .net winforms garbage-collection

我继承了一个WinForms项目,并且遇到了一个奇怪的问题,我已经将其缩小为GC问题。我根本看不出它是怎么回事,但随意告诉我: - )

每隔几个月申请一次。我无法定期复制它,但它已经发生在我的客户几次,而且我在开发人员环境中看到过两次(但从来没有在我寻找它时)。

以下代码简化了设置。

public class Main
{
  Main()
  {
    var listOfB = new List<ObjB>();
    var objB = new ObjB();
    listOfB.Add(objB);
    var objA = new ObjA(objB);

    // Do alot of stuff with listOfB and different lists of objA-like-objects.
    // Sometimes objA only exists as objects in a DataGrid

    // listOfB exists for a long time, while objA comes and goes..
  }
}

class ObjA
{
  private bool _isBNull;
  private ObjB _objB;

  public ObjA(ObjB objB)
  {
    _objB = objB;
    _isBNull = objB == null;
  }

  public ObjB ObjB
  {
    get
    {
      if (!_isBNull && _objB == null)
      {
        //
        // This should NEVER happen, but happens anyway :-(
        //

        // Do some logging and now that we 
        // know it CAN happen anyway we throw up...
      }
      return _objB;
    }
    set
    {
      _isBNull = value == null;
      _objB = value;
    }
  }
}

class ObjB
{
}

我们最终处于objB为null的应该是不可能的条件,但是我们的debug-track-boolean告诉我们它不应该是。

有没有人经历过这样的事情?关于做什么或在哪里看的任何建议?

该项目是.Net 4.0,并且在32位XP和64位Windows 7客户端计算机上看到了错误。

Smøller - 一个WinForm新手

1 个答案:

答案 0 :(得分:5)

您的问题出在您的设置器中,而不是在gc中:

set
{
  _isBNull = value == null;
  _objB = value;
}

setter / getter不是线程安全的,如果你有多个线程同时访问getter和setter你可以得到一个位置,其中_isBNull == null而_objB是一个对象。

在查看您当前的代码时,我认为没有任何理由拥有单独的属性_isBNull,您必须做出的修改,这取决于您对该类的功能要求。