c# – 处理对象的代码分析友好方式

作为Visual Studio 2010(主要是C#4.0)开发标准的一部分,我们启用了代码分析.当我正在审查最近为新项目提交的代码时,我看到了很多

CA2000 : Microsoft.Reliability: In method ‘XYZ’,object ‘ABC’ is not
disposed along all exception paths. Call System.Idisposable.dispose on
object ‘ABC’ before all references to it are out of scope.

警告.问题是,我所做的一切似乎都没有消除警告 – 我花了几个小时搜索网络并尽我所能.

首先,让我明确一点,我不是在讨论如何使用简单的使用块来正确处理局部变量 – 这不是问题.在我的情况下,当对象由方法返回或分配给方法中的另一个对象时,会出现这些警告.

这是一个代码示例,其中包含四个此类警告:

public void MainMethod()
{
    var object1 = CreateFirstObject();    // Warning here
    var object2 = CreateSecondobject();   // Warning here

    SomeCollectionProperty.Add(object1);
    SomeCollectionProperty.Add(object2);
}

private SomeObject CreateFirstObject()
{
    var theObject = new SomeObject()      // Warning here
    {
        FirstProperty = "some value",// ...
    };

    return theObject;
}

private SomeOtherObject CreateSecondobject()
{
    var theObject = new SomeOtherObject() // Warning here
    {
        FirstProperty = "a different value",// ...
    };

    return theObject;
}

我已经评论了警告发生的地方.

我已经尝试重构两个Create方法,如MSDN文章(here)中所述,但警告仍然出现.

UPDATE
我应该注意SomeObject和SomeOtherObject都实现了Idisposable.

此外,尽管对象初始值设定项可能是问题的一个组成部分,但请记住,初始化程序与两个私有方法隔离,与MainMethod中的警告无关.

谁能告诉我如何正确实施这些方法来消除CA2000警告?

解决方法

在这种情况下,CA2000正在检测的问题是,如果在将异常传递出方法之前发生异常,则一次性实例可能是“孤立的”.例如,CreateFirstObject的“正确”实现看起来如下所示:
private SomeObject CreateFirstObject()
{
    var theObject = new SomeObject();
    try
    {
        theObject.FirstProperty = "some value";
    }
    catch
    {
        theObject.dispose();
        throw;
    }

    return theObject;
}

鉴于您所描述的有关MainMethod所需行为的内容,其“正确”实现可能如下所示:

public void MainMethod()
{
    var object1 = CreateFirstObject();
    try
    {
        SomeCollectionProperty.Add(object1);

        var object2 = CreateSecondobject();
        try
        {
            SomeCollectionProperty.Add(object2);
        }
        catch
        {
            object2.dispose();
            throw;
        }
    }
    catch
    {
        object1.dispose();
        SomeCollectionProperty.Remove(object1); // Not supposed to throw if item does not exist in collection.

        throw;
    }
}

相关文章

在要实现单例模式的类当中添加如下代码:实例化的时候:frmC...
1、如果制作圆角窗体,窗体先继承DOTNETBAR的:public parti...
根据网上资料,自己很粗略的实现了一个winform搜索提示,但是...
近期在做DSOFramer这个控件,打算自己弄一个自定义控件来封装...
今天玩了一把WMI,查询了一下电脑的硬件信息,感觉很多代码都...
最近在研究WinWordControl这个控件,因为上级要求在系统里,...