草庐IT

c++ - 工厂设计模式如何处理不同长度的构造函数?

coder 2024-02-25 原文

我想要一个类,它根据我传递给它的字符串创建不同种类的对象。根据我的研究,这最能描述工厂设计模式。我成功地实现了它,但我遇到了一个设计问题:我不知道如何创建具有不同长度构造函数的对象。

让我们举个例子,一个叫做Pet的抽象父类。其中有 3 个 child :鱼、猫和狗。它们都从 Pet 继承了重量和颜色,因此它们的构造函数中也有。但是一条鱼可能需要一些鳍和一个关于它是否是咸水鱼的 bool 值。这是一个 4 参数构造函数。猫想要腿的数量。那是3个参数。这只狗可能有腿、品种以及他是否与其他狗玩得好等参数,共 5 个参数。

在 C++ 中,我知道没有任何反射,所以最常见的做法似乎是只声明一个字符串到函数指针的映射,其中函数指针指向一个看起来像这样的函数:

    template<typename T> Pet* createObject(int weight, std::string color) {return new T(weight, color);}

同样,我不确定如何在调用中填充更多参数而不影响其他对象的构造函数的调用。

我可以想到两种解决方法:创建新函数以接受不同数量的参数,或者为超过一定大小的构造函数设置默认参数。

解决方法 1 似乎有点过分,具体取决于我有多少不同的参数大小。

解决方法 2 似乎忽略了构造函数的整个要点,因为我将在调用构造函数后被迫分配数据。

还有其他更好的解决方法吗?

最佳答案

您可以使用可变参数模板和完美转发。

template<typename T, typename... Args>
Pet* createObject(Args&&... args) {
    return new T(std::forward<Args>(args)...);
}

但是,由于任何指针都可以转换为其基类,因此如果此函数返回 T* 可能会更好。此外,使用裸指针并不明智,因为您必须手动删除它们。最好使用shared_ptrunique_ptr .对于这些类,已经有类似的工厂方法:make_sharedmake_unique (后者仅在 C++14 中)。或者,如果您的编译器不支持 C++11,那么您可以使用 Boost 中的 shared_ptrmake_shared .

当然,当您在编译时知道需要创建什么类型时,此解决方案就可以使用。如果你必须在运行时决定它,那么整个问题必须从不同的方向来考虑,就像你不知道你要创建什么类型,那么你就没有办法知道给他们什么参数,所有类型通用的参数除外。在这种情况下,您需要的是 abstract factory pattern .幸运的是,C++(至少从 C++11 开始)提供了一种无需创建大量类即可实现此模式的方法。例如,假设您必须创建从 Pet 派生的某个类的实例。宠物的实际种类、大小和其他属性在其他地方决定,而宠物的名称在创建时决定。然后,您需要一个这样的工厂:

typedef std::function<std::shared_ptr<Pet>(const std::string& name)> PetFactory;

在某些时候,您决定要创建一个Dog(我将实际创建参数的含义留给您自己想象)。

PetFactory petFactory =
        [](const std::string& name) {
            return std::make_shared<Dog>(name, 12, "brown", 23.5);
        }

当你真正创建它时,你只需要调用工厂:

std::shared_ptr<Pet> pet = petFactory("Pet Name");

关于c++ - 工厂设计模式如何处理不同长度的构造函数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31477785/

有关c++ - 工厂设计模式如何处理不同长度的构造函数?的更多相关文章

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

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

  2. ruby-on-rails - Railstutorial : db:populate vs. 工厂女孩 - 2

    在railstutorial中,作者为什么选择使用这个(代码list10.25):http://ruby.railstutorial.org/chapters/updating-showing-and-deleting-usersnamespace:dbdodesc"Filldatabasewithsampledata"task:populate=>:environmentdoRake::Task['db:reset'].invokeUser.create!(:name=>"ExampleUser",:email=>"example@railstutorial.org",:passwo

  3. 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

  4. ruby-on-rails - 如何优雅地重启 thin + nginx? - 2

    我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server

  5. 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

  6. ruby - 在没有 sass 引擎的情况下使用 sass 颜色函数 - 2

    我想在一个没有Sass引擎的类中使用Sass颜色函数。我已经在项目中使用了sassgem,所以我认为搭载会像以下一样简单:classRectangleincludeSass::Script::FunctionsdefcolorSass::Script::Color.new([0x82,0x39,0x06])enddefrender#hamlengineexecutedwithcontextofself#sothatwithintemlateicouldcall#%stop{offset:'0%',stop:{color:lighten(color)}}endend更新:参见上面的#re

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

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

  8. ruby-on-rails - Enumerator.new 如何处理已通过的 block ? - 2

    我在理解Enumerator.new方法的工作原理时遇到了一些困难。假设文档中的示例:fib=Enumerator.newdo|y|a=b=1loopdoy[1,1,2,3,5,8,13,21,34,55]循环中断条件在哪里,它如何知道循环应该迭代多少次(因为它没有任何明确的中断条件并且看起来像无限循环)? 最佳答案 Enumerator使用Fibers在内部。您的示例等效于:require'fiber'fiber=Fiber.newdoa=b=1loopdoFiber.yieldaa,b=b,a+bendend10.times.m

  9. ruby-on-rails - 在 ruby​​ 中使用 gsub 函数替换单词 - 2

    我正在尝试用ruby​​中的gsub函数替换字符串中的某些单词,但有时效果很好,在某些情况下会出现此错误?这种格式有什么问题吗NoMethodError(undefinedmethod`gsub!'fornil:NilClass):模型.rbclassTest"replacethisID1",WAY=>"replacethisID2andID3",DELTA=>"replacethisID4"}end另一个模型.rbclassCheck 最佳答案 啊,我找到了!gsub!是一个非常奇怪的方法。首先,它替换了字符串,所以它实际上修改了

  10. ruby - 在 Ruby 中有条件地定义函数 - 2

    我有一些代码在几个不同的位置之一运行:作为具有调试输出的命令行工具,作为不接受任何输出的更大程序的一部分,以及在Rails环境中。有时我需要根据代码的位置对代码进行细微的更改,我意识到以下样式似乎可行:print"Testingnestedfunctionsdefined\n"CLI=trueifCLIdeftest_printprint"CommandLineVersion\n"endelsedeftest_printprint"ReleaseVersion\n"endendtest_print()这导致:TestingnestedfunctionsdefinedCommandLin

随机推荐