参考滥用:值得清理?
我inheritance了一些广泛且不必要地使用ref关键字的代码。 如果不使用ref ,原始开发人员显然担心对象会被克隆为原始类型,并且在编写50k +代码行之前没有费心去研究这个问题。
这与其他不良编码实践相结合,已经创造了一些表面上非常危险的情况。 例如:
Customer person = NextInLine(); //person is Alice person.DataBackend.ChangeAddress(ref person, newAddress); //person could now be Bob, Eve, or null
你能想象走进商店改变你的地址,走出一个完全不同的人吗?
可怕,但在实践中,在这个应用程序中使用ref似乎是无害的多余。 我无法certificate清理它需要花费大量时间。 为了帮助推销这个想法,我提出了以下问题:
如何不必要地使用ref是破坏性的?
我特别关心维护。 具有示例的合理答案是优选的。
我们也欢迎你争辩说没有必要进行清理。
我想说最大的危险是如果参数在函数内部由于某种原因设置为null
:
public void MakeNull(ref Customer person) { // random code person = null; return; }
现在,你不仅仅是一个与众不同的人,你已经完全被淘汰了!
只要开发此应用程序的人都理解:
默认情况下,对象引用按值传递。
和:
使用
ref
关键字,对象引用通过引用传递。
如果代码现在按预期工作,并且您的开发人员了解其中的差异,则可能不值得将其全部删除。
我将添加我见过的ref关键字的最坏用法,该方法看起来像这样:
public bool DoAction(ref Exception exception) {...}
是的,您必须声明并传递一个Exception引用才能调用该方法,然后检查该方法的返回值,以查看是否已捕获并通过refexception返回exception。
你能弄清楚为什么代码的发起者认为他们需要将参数作为参考吗? 是因为他们确实更新了它然后删除了function,还是因为他们当时不理解c#?
如果你认为清理是值得的,那么继续吧 – 特别是如果你现在有时间的话。 当出现真正的问题时,您可能无法正确地修复它,因为它很可能是一个紧急的错误修复,您将没有时间做正确的工作。
在C#中修改方法中参数的值是很常见的,因为它们通常是按值而不是由ref。 这适用于参考和价值类型; 例如,将引用设置为null将更改原始引用。 当其他开发人员像往常一样工作时,这可能会导致非常奇怪和痛苦的错误。 使用ref参数创建递归方法是不行的。
除此之外,您对ref可以传递的内容有限制。 例如,您不能传递常量值,只读字段,属性等,因此在使用ref参数调用方法时需要许多辅助变量。
最后但并非最不重要的是性能如果可能不太好,因为它需要更多间接(ref只是一个需要在每次访问时解决的参考)并且还可以使对象保持更长时间,因为引用不会出现范围很快。
对我来说,闻起来就像一个C ++开发人员做出了无根据的假设。
我会谨慎地改变那些有效的东西。 (我假设它有效,因为你没有评论它被破坏,只是它很危险)。
你要做的最后一件事是打破一些微妙的东西,并且必须花一周的时间来追踪问题。
我建议你随时清理 – 一次一种方法。
- 找到一个使用ref的方法,确定它不是必需的。
- 更改方法签名并修复调用。
- 测试。
- 重复。
虽然您遇到的具体问题可能比大多数情况更严重,但您的情况非常普遍 – 拥有大量的代码库,不符合我们目前对“ 正确方法 ”的理解。
批发“升级”经常遇到困难。 随你进行重构 – 在处理它们时提高规格 – 更加安全。
这是建筑行业的先例。 例如,除非出现问题,否则不需要触摸较旧(例如,19世纪)建筑物中的电线。 但是,当出现问题时,必须按照现代标准完成新工作。
我会尝试解决它。 只需使用正则表达式执行解决方案宽字符串替换,然后检查unit testing。 我知道这可能会破坏代码。 但是你多久使用一次参考? 几乎从不,对吧? 鉴于开发人员不知道它是如何工作的,我认为它在某个地方(它应该的方式)使用的可能性更小。 如果代码中断 – 好吧,回滚……
如何不必要地使用ref是破坏性的?
其他答案涉及语义问题,这绝对是最重要的事情。 好的代码是自我记录的,当我给出一个ref参数时,我认为它会改变。 如果没有,则API无法自我记录。
但为了好玩,我们如何看待另一个方面 – 性能?
void ChangeAddress(ref Customer person, Address address) { person.Address = address; }
这里person
是对引用的引用,因此每当您访问它时都会引入一些间接。 让我们看看这可能转化为的一些组件:
mov eax, [person] ; load the reference to person. mov [eax+Address], address ; assign address to person.Address.
现在非ref版本:
void ChangeAddress(Customer person, Address address) { person.Address = address; }
这里没有间接,所以我们可以摆脱一个阅读:
mov [person+Address], address ; assign address to person.Address.
在实践中,人们希望.NET在ref
版本中缓存[person]
,在多次访问中分摊间接成本。 实际上,除了像这里的琐碎方法之外,指令数量实际上不会下降50%。
上述就是C#学习教程:参考滥用:值得清理?分享的全部内容,如果对大家有所用处且需要了解更多关于C#学习教程,希望大家多多关注—计算机技术网(www.ctvol.com)!
本文来自网络收集,不代表计算机技术网立场,如涉及侵权请联系管理员删除。
ctvol管理联系方式QQ:251552304
本文章地址:https://www.ctvol.com/cdevelopment/986738.html