c# - 在不破坏向后兼容性的情况下更改参数名称

标签 c# backwards-compatibility named-parameters

我正在开发一个 C# 库,所以我们担心破坏向后兼容性,但我想知道是否可以只更改参数的名称并保持向后兼容性,因为能够使用命名参数?下面是我正在尝试做的一个例子

[Obsolete("use ChangeSpecificFoo(SpecificFoo specificFoo)")]
public void ChangeSpecificFoo(SpecificFoo foo)
{
    _specificFoo = foo;
}

//Compile error ... already defines a member called 'ChangeSpecificFoo' with the same parameter types
public void ChangeSpecificFoo(SpecificFoo specificFoo)
{
    _specificFoo = specificFoo;
}

仅更改参数名称会带来破坏向后兼容性的潜在风险,因为有人可能会使用命名参数调用该方法,例如 ChangeSpecificFoo(foo: someSpecificFoo) ,但我们不能通过以下方式弃用该方法添加具有正确参数名称的新方法,因为参数名称未包含在方法签名中,因此编译器将其视为重复项。

有什么办法解决这个问题吗?我看到的唯一替代方法是更改​​方法名称,使其不重复,然后弃用旧方法,或者等到我们从参数列表中添加或删除参数,然后更改参数名称(这可能永远不会发生,因为该方法是相当稳定),或者只是进行更改并修复我们发现的使用此库的代码可能遇到的任何中断。

最佳答案

我的第一个倾向很简单:不要。您的参数名称在方法主体之外无关紧要。您认为人们会直呼其名,因此可能会破坏它,这是对的。但是,仅更改参数的名称并没有真正的好处。

更改名称的唯一可能原因是重新定义方法的作用,因为旧名称会导致混淆。在那种情况下,方法的名称也应该更改,以免引入另一种形式的混淆。 (方法签名相同的事实是不更改参数名称的第一个也是更重要的原因。但是,这可能会解释为什么您可能想要这样做。)

但是,如果您仍然坚持保留相同的方法签名,但更改名称,则可以这样做。 (再次,我强烈建议您要么根本不更改它,要么重命名该方法以继续消除混淆。)

解决此问题的一种方法是使该方法具有两个参数,但使第二个参数可选。让最后一个参数使用旧名称,然后在方法中将其分配给它。

我还强烈建议记录命名参数的任何使用,以查看您对人们将其称为命名参数的担忧是否有效。

public void ChangeSpecificFoo(SpecificFoo specificFoo = null, SpecificFoo foo = null)
{
    if (foo != null && specificFoo == null)
    {
        // Add any other details you can, especially 
        // to figure out who is calling this.
        Log("Someone used a name parameter!!");
    }
    _specificFoo = specificFoo ?? foo;
}

作为Dmitry Bychenko在评论中指出,这不会阻止任何人像这样调用此方法:ChangeSpecificFoo(null, new SpecificFoo()),这将触发日志记录。

他的观察介绍了为什么这是一个坏主意的另一个原因:您现在正在引入另一种方式让人们“错误地”调用您的方法。因此,我将重复我的回答开头的建议:不要这样做,除非您真的真的需要更改该参数名称。

关于c# - 在不破坏向后兼容性的情况下更改参数名称,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38058190/

相关文章:

c# - 是否可以选择 Windows 窗体标签上的文本?

c# - 调用线程无法访问此对象,因为另一个线程拥有它 - BackgroundWorker 错误

c# - 在 C# 中退出阻塞 TCP 读取的线程

html - HTML在构造参数时遵循什么规则?

methods - 为什么函数调用需要 Swift 中的参数名称?

c# - 在变量值发生变化时调用函数的最佳方式?

python - 如何在 Windows 中同时安装 Python 2.x 和 Python 3.x

android - 替换 fragment 和方向更改

javascript - 匀场映射/减少/过滤器在 Exploder 6/7/8 中工作吗?

python - 使用 SQLAlchemy 的数据库函数中的命名参数