2

假设我正在编写一个实用程序库,它定义了一个具有两个重载的方法:

public static class MyClass
{
    public static void DoSomething(string myValue, bool myFlag)
    {
        Contract.Requires<ArgumentNullException>(myValue != null);
        CallExternalMethod(myValue);
        if(myFlag){
            AlsoCallOtherMethod(myValue);
        }        
    }

    public static void DoSomething(string myValue)
    {
        Contract.Requires<ArgumentNullException>(myValue != null);
        DoSomething(myValue, true);
    }
}

作为最佳实践,我只在一种方法中定义逻辑,并使用重载来指定默认参数(.Net 3.5,我不能使用 .Net 4 默认值参数)。

如您所见,我还通过使用合同来验证输入。

由于逻辑只在第一种方法中,那么第二个 Contract 没用吗?

关于运行时检查,我知道它没用,但是静态检查器呢?理解我的模式是否足够聪明?

4

1 回答 1

2

我完全不同意BonyT 的回答

代码契约是您的文档和方法契约的一部分,就像类型和名称一样。用户/开发人员没有理由假设您只是调用其他方法重载而没有别的。找出哪些合同适用于方法不应该是一个猜谜游戏。如果对参数或返回值有约束,请使用代码合同对其进行记录,即使您在内部 - 对其他人隐藏 - 只需调用另一个方法。

你不能把代码契约仅仅看作一个断言工具:控制流运行到另一个重载,它检查参数,所以我不必在这里做。您必须将其视为文档。

在实际层面上:添加代码合同也会将这些合同添加到您生成的 XML 文档中。静态检查器和代码合同编辑器扩展等工具可以向用户显示合同。在 Pascal Cuoq 的这篇相关文章Jon Skeet 的这篇文章中可以找到一些其他好的论点。

但是,从运行时的角度来看,它多余的。至于运行时开销:大多数检查将相对较小,并且没有性能问题。一些检查可能与性能有关(尤其是对集合、数组和枚举的检查),但可以在项目的代码合同属性对话框中单独禁用这些检查(例如,对于发布版本)。一般来说,合同也是如此:您可以将检查降低到您想要的任何级别,以便在非常关键的时候提高性能。如果您构建合同程序集,用户仍将获得静态检查。

总结一下。指定冗余 Requires 和 Ensures 的缺点:

  • 相同的条件可能会检查两次或更多次
  • 由于多次检查(如果您不禁用它)导致的微小的负面性能影响
  • 更多打字(但请参阅这些片段

专业人士:

  • 代码文档
  • 生成的 XML 中的文档
  • 代码用户的静态检查器按预期工作
  • 代码合同编辑器扩展工作
  • ArgumentExceptionfor Requires 发生在被调用的方法中,而不是在远处的某个方法中
于 2012-07-27T12:01:38.540 回答