草庐IT

c# - 对于封装和可重用性,首选扩展方法?

coder 2023-06-03 原文

edit4:讲究了,因为这似乎更多地变成了讨论,而不是特定的问题。
在C++编程中,通常最好的做法是“首选非成员非友函数”而不是实例方法。 Scott Meyers在this classic Dr. Dobbs article中推荐了此方法,Herb Sutter和Andrei Alexandrescu在C++ Coding Standards中重复了这一点(项目44);一般的论点是,如果一个函数可以仅依靠类公开的公共(public)接口(interface)来完成其工作,则实际上它增加了封装使其具有外部性。虽然这在某种程度上使类的“打包”感到困惑,但通常认为这样做是值得的。
现在,自从我开始使用C#编程以来,我一直感觉到在这里是他们试图通过“属于非成员,非 friend 功能的一部分”实现的概念的最终表达。类接口(interface)”。 C#向混合添加了两个关键组件-第一个是interfaces,第二个是extension methods:

  • 接口(interface)允许类正式指定其公共(public)合约,它们要向世人展示的方法和属性。
  • 任何其他类都可以选择实现相同的接口(interface)并履行相同的契约(Contract)。
  • 扩展方法可以在接口(interface)上定义,向所有实现者自动提供可以通过该接口(interface)实现的任何功能。
  • 最重要的是,由于有“实例语法”糖和IDE支持,因此可以像调用其他任何实例方法一样调用它们,从而消除了认知开销!

  • 因此,借助成员的便利性,您可以获得“非成员,非 friend ”功能的封装优势。在我看来,两全其美。 .NET库本身在LINQ中提供了一个出色的示例。但是,在我看来,到处都可以看到有人警告不要扩展方法。甚至MSDN页面本身也指出:

    In general, we recommend that you implement extension methods sparingly and only when you have to.


    (编辑:)即使在当前的.NET库中,我也可以看到具有扩展而不是实例方法的扩展很有用的地方-例如,List<T>的所有实用功能(SortBinarySearchFindIndex等。)如果将其提升为IList<T>,将非常有用-获得类似的免费红利功能,将为实现该接口(interface)带来更多好处。)
    那么判决是什么?扩展方法是封装和代码重用的顶峰,还是我只是在欺骗自己?
    ( edit2:)作为对Tomas的回应-尽管C#最初是从Java的OO心态开始的,但似乎每个新版本都采用了更多的多范式编程;这个问题的主要目的是是否使用扩展插入样式改变的方法(朝着更通用/功能更强的C#)有用或值得。
    edit3:可覆盖的扩展方法
    到目前为止,使用这种方法唯一真正的问题是,如果需要的话,您不能专门研究扩展方法。我一直在考虑这个问题,并且我想出了一个解决方案。
    假设我有一个接口(interface)MyInterface,我想扩展它-
    我在MyExtension静态类中定义扩展方法,并将其与另一个接口(interface)配对,称为MyExtensionOverriderMyExtension方法是根据以下模式定义的:
    public static int MyMethod(this MyInterface obj, int arg, bool attemptCast=true)
    {
        if (attemptCast && obj is MyExtensionOverrider)
        {
            return ((MyExtensionOverrider)obj).MyMethod(arg);
        }
        // regular implementation here
    }
    
    覆盖接口(interface)反射(reflect)了MyExtension中定义的所有方法,但没有thisattemptCast参数的情况除外:
    public interface MyExtensionOverrider
    {
        int MyMethod(int arg);
        string MyOtherMethod();
    }
    
    现在,任何类都可以实现该接口(interface)并获得默认的扩展功能:
    public class MyClass : MyInterface { ... }
    
    任何想要通过特定实现覆盖它的人都可以实现覆盖接口(interface):
    public class MySpecializedClass : MyInterface, MyExtensionOverrider
    {
        public int MyMethod(int arg) 
        { 
            //specialized implementation for one method
        }
        public string MyOtherMethod() 
        {   // fallback to default for others
            MyExtension.MyOtherMethod(this, attemptCast: false); 
        }
    }
    
    然后我们开始:接口(interface)上提供的扩展方法,并在需要时选择完全可扩展的选项。同样完全通用,接口(interface)本身也不需要知道扩展名/替代项,并且可以实现多个扩展名/替代对而不会互相干扰。
    我可以看到这种方法的三个问题-
  • 有点脆弱-扩展方法和重写接口(interface)必须手动保持同步。
  • 有点丑陋-实现覆盖接口(interface)涉及到您不想专门化的每个功能的样板。
  • 有点慢-在每个方法的主线中都有一个额外的bool比较和强制转换尝试。

  • 尽管如此,尽管如此,我认为这是我们能够获得的最好的解决方案,直到对接口(interface)功能的语言支持为止。有什么想法吗?

    最佳答案

    我通常喜欢扩展方法,尤其是在接口(interface)上,但是它们有两个问题:

    首先,如果实现具有实现扩展方法目的的更有效方法,则没有通用的表达方式。例如,Enumerable.Count()明确知道ICollection/ICollection<T>并对其进行特殊处理。一种替代方法是,如果接口(interface)实际上可以直接包含实现,而仅引用其他接口(interface)方法而不声明字段。然后可以在适当的实现中覆盖这些方法。当然,这确实意味着您需要拥有接口(interface)。但是在某些情况下,它比当前的扩展方法更干净。 (通过避免引入字段的能力,我相信您会遇到一些实现问题,这些问题会引入类的多重继承。)

    第二,我不喜欢扩展方法的发现方式。不能说“我想要类X的扩展方法”而不用将同一 namespace 中其他类的扩展方法拖入。我希望你能写:

    using static System.Linq.Enumerable;
    

    只选择那些扩展方法。

    (顺便说一句,我将在周四的NDC 2010上更多地讨论这两个方面。希望这次谈话能被记录下来。)

    指定仅依赖公共(public)接口(interface)的通用算法的能力很好。在提供接口(interface)的类型上调用这些算法的能力很好。当前的机制只是有一些尖锐的角落。

    顺便说一句,能够在一个类型中编写方法,但是说:“限制我只能使用公共(public)API”,这可能会很好。

    关于c# - 对于封装和可重用性,首选扩展方法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3037250/

    有关c# - 对于封装和可重用性,首选扩展方法?的更多相关文章

    1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

      我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

    2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

      总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

    3. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

      类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

    4. ruby - Facter::Util::Uptime:Module 的未定义方法 get_uptime (NoMethodError) - 2

      我正在尝试设置一个puppet节点,但ruby​​gems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由ruby​​gems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby

    5. Ruby 方法() 方法 - 2

      我想了解Ruby方法methods()是如何工作的。我尝试使用“ruby方法”在Google上搜索,但这不是我需要的。我也看过ruby​​-doc.org,但我没有找到这种方法。你能详细解释一下它是如何工作的或者给我一个链接吗?更新我用methods()方法做了实验,得到了这样的结果:'labrat'代码classFirstdeffirst_instance_mymethodenddefself.first_class_mymethodendendclassSecond使用类#returnsavailablemethodslistforclassandancestorsputsSeco

    6. ruby-on-rails - Rails 3.2.1 中 ActionMailer 中的未定义方法 'default_content_type=' - 2

      我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer

    7. ruby - 使用 C 扩展开发 ruby​​gem 时,如何使用 Rspec 在本地进行测试? - 2

      我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当

    8. ruby - Highline 询问方法不会使用同一行 - 2

      设置:狂欢ruby1.9.2高线(1.6.13)描述:我已经相当习惯在其他一些项目中使用highline,但已经有几个月没有使用它了。现在,在Ruby1.9.2上全新安装时,它似乎不允许在同一行回答提示。所以以前我会看到类似的东西:require"highline/import"ask"Whatisyourfavoritecolor?"并得到:Whatisyourfavoritecolor?|现在我看到类似的东西:Whatisyourfavoritecolor?|竖线(|)符号是我的终端光标。知道为什么会发生这种变化吗? 最佳答案

    9. ruby - 主要 :Object when running build from sublime 的未定义方法 `require_relative' - 2

      我已经从我的命令行中获得了一切,所以我可以运行rubymyfile并且它可以正常工作。但是当我尝试从sublime中运行它时,我得到了undefinedmethod`require_relative'formain:Object有人知道我的sublime设置中缺少什么吗?我正在使用OSX并安装了rvm。 最佳答案 或者,您可以只使用“require”,它应该可以正常工作。我认为“require_relative”仅适用于ruby​​1.9+ 关于ruby-主要:Objectwhenrun

    10. ruby - 多个属性的 update_column 方法 - 2

      我有一个具有一些属性的模型:attr1、attr2和attr3。我需要在不执行回调和验证的情况下更新此属性。我找到了update_column方法,但我想同时更新三个属性。我需要这样的东西:update_columns({attr1:val1,attr2:val2,attr3:val3})代替update_column(attr1,val1)update_column(attr2,val2)update_column(attr3,val3) 最佳答案 您可以使用update_columns(attr1:val1,attr2:val2

    随机推荐