草庐IT

php - 设计模式——反模式调用超——避免或保留

coder 2024-04-16 原文

我最近注意到在某些情况下我真正喜欢做的事情(调用 super)是一种反模式。 ( http://en.wikipedia.org/wiki/Call_super )

所以我的问题是:

How to do the following thing(我们大多数人都需要的一个基本的东西,给一个对象添加一些细节)。我将向保存在对象 var 中的数组添加一个新元素:

   <?php
   class A {
       // bla bla bla
       public function __construct() {
            $this->_data['newDetail'] = "Ipiicaei";
       }
   }


   class B extends A {
       // bla bla bla
       // Override constructor, I need to add one more detail for this class
       public function __construct() {
            parent::__construct();
            $this->_data['newDetailSubcls'] = "Something";
       }
   }

现在...如果我不调用父类构造函数来添加第一个元素,我有两种可能性:

  1. 在每个子类中添加父类通常执行的代码部分,并将其从父类中删除。因此,如果我用 999 个类扩展它,我将在父类中复制 999*行。对我来说听起来很糟糕。

  2. 从父级调用在子级中实现的方法(模板方法模式,他们推荐的)。因此,如果我只需要一个类来添加它,而 998 的其余部分的行为与父类完全一样,那么我在每个类中添加一个空函数。这对我来说也很糟糕。

请记住,我的示例很简单(示例应该如此),但父类和/或子类可能会做复杂的事情。

我明白为什么调用 super 在某些情况下可能不好的原因。但在我看来,这个还可以。

那么...您将如何处理?忽略 call super 是一种反模式,并像我一样做(或者如果我的方式不好,我以前会这样做)?或者...如何?

最佳答案

使用构造函数是这种反模式如何出现的一个不好的例子,因为在 PHP 中就像大多数支持继承的语言一样,

If the child does not define a constructor then it may be inherited from the parent class just like a normal class method ... (from the PHP manual).

如果您尝试定义的其他 998 个类中的任何一个有意不包含构造函数,则将调用基类构造函数,这使得几乎需要在每个派生类上添加一个构造函数 来避免这个功能。这就是为什么不建议在构造函数中添加可能改变 protected 字段/属性状态的逻辑或调用可能在派生类中被覆盖的虚拟方法的原因。

然而,反模式实际上表达了:

Note that it is the requirement of calling the parent that is the anti-pattern (from Wikipedia)

因此,只要派生类不需要调用方法的基类版本就可以。即使从未调用基方法,派生类也必须正确运行。这将避免派生类重写虚方法但未实现某些特殊要求(可能是因为基类缺少文档)的情况,此时派生类的行为不符合系统预期,可能导致系统崩溃或出现意外行为,这是对 Liskov substitution principle 的典型违反.

在您的特定示例中,此反模式不适用:您正在扩展构造函数,根据定义,在将类实例化为有用状态时需要调用构造函数。在大多数情况下,这就是您想要的。

关于php - 设计模式——反模式调用超——避免或保留,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17150357/

有关php - 设计模式——反模式调用超——避免或保留的更多相关文章

  1. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

    我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

  2. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  3. ruby - 如何在续集中重新加载表模式? - 2

    鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

  4. ruby-on-rails - 使用 rails 4 设计而不更新用户 - 2

    我将应用程序升级到Rails4,一切正常。我可以登录并转到我的编辑页面。也更新了观点。使用标准View时,用户会更新。但是当我添加例如字段:name时,它​​不会在表单中更新。使用devise3.1.1和gem'protected_attributes'我需要在设备或数据库上运行某种更新命令吗?我也搜索过这个地方,找到了许多不同的解决方案,但没有一个会更新我的用户字段。我没有添加任何自定义字段。 最佳答案 如果您想允许额外的参数,您可以在ApplicationController中使用beforefilter,因为Rails4将参数

  5. 使用 ACL 调用 upload_file 时出现 Ruby S3 "Access Denied"错误 - 2

    我正在尝试编写一个将文件上传到AWS并公开该文件的Ruby脚本。我做了以下事情:s3=Aws::S3::Resource.new(credentials:Aws::Credentials.new(KEY,SECRET),region:'us-west-2')obj=s3.bucket('stg-db').object('key')obj.upload_file(filename)这似乎工作正常,除了该文件不是公开可用的,而且我无法获得它的公共(public)URL。但是当我登录到S3时,我可以正常查看我的文件。为了使其公开可用,我将最后一行更改为obj.upload_file(file

  6. ruby-on-rails - RSpec:避免使用允许接收的任何实例 - 2

    我正在处理旧代码的一部分。beforedoallow_any_instance_of(SportRateManager).toreceive(:create).and_return(true)endRubocop错误如下:Avoidstubbingusing'allow_any_instance_of'我读到了RuboCop::RSpec:AnyInstance我试着像下面那样改变它。由此beforedoallow_any_instance_of(SportRateManager).toreceive(:create).and_return(true)end对此:let(:sport_

  7. c# - 如何在 ruby​​ 中调用 C# dll? - 2

    如何在ruby​​中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL

  8. java - 从 JRuby 调用 Java 类的问题 - 2

    我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

  9. ruby - 调用其他方法的 TDD 方法的正确方法 - 2

    我需要一些关于TDD概念的帮助。假设我有以下代码defexecute(command)casecommandwhen"c"create_new_characterwhen"i"display_inventoryendenddefcreate_new_character#dostufftocreatenewcharacterenddefdisplay_inventory#dostufftodisplayinventoryend现在我不确定要为什么编写单元测试。如果我为execute方法编写单元测试,那不是几乎涵盖了我对create_new_character和display_invent

  10. ruby - 是否有用于序列化和反序列化各种格式的对象层次结构的模式? - 2

    给定一个复杂的对象层次结构,幸运的是它不包含循环引用,我如何实现支持各种格式的序列化?我不是来讨论实际实现的。相反,我正在寻找可能会派上用场的设计模式提示。更准确地说:我正在使用Ruby,我想解析XML和JSON数据以构建复杂的对象层次结构。此外,应该可以将该层次结构序列化为JSON、XML和可能的HTML。我可以为此使用Builder模式吗?在任何提到的情况下,我都有某种结构化数据-无论是在内存中还是文本中-我想用它来构建其他东西。我认为将序列化逻辑与实际业务逻辑分开会很好,这样我以后就可以轻松支持多种XML格式。 最佳答案 我最

随机推荐