鉴于以下预先存在的框架,我需要找到好的设计模式来创建派生类的不同实例。
我面临的主要挑战如下:
challenge-1> 每个类都有10多个字段,如何有效地将这些字段传递给派生类,再传递给基类。
针对这个问题,我可以想出四个解决方案,但没有一个对我有吸引力。
方法一>以简单格式传递所有参数
classA::classA(int field1, float field2, ..., double field29)
=> 缺点:创建传入参数超过 6~7 个的函数不是一个好主意
方法2>将所有参数作为结构传递
struct DataClassA
{
int field1;
float field2;
...
double field29;
};
struct DataClassBA : DataClassA
{
int m_iField30;
// ...
double m_iField40;
};
所以首先我将 DataClassBA 传递给 classBA 然后 classBA 将 DataClassA 传递给 classA 。
=> 缺点:类型 DataClassBA 和 classBA 是相似的类型,除了一个包含操作而另一个不包含操作。此外,当将结构传递给构造函数时,存在复制和重复的惩罚。想象一下,对于每一个不同的类,我们必须定义一个相似的结构来保存所有的初始化数据。类与其对应结构的主要区别在于类包含一些方法,而结构纯粹用于传输数据。
方法三>通过Set函数设置所有字段
classA
{
public:
int Field1() const { return m_iField1; }
classA& Field1(int field1)
{
m_iField1 = field1;
return *this;
}
...
}
classBA : public classA
{
public:
int Field30() const { return m_iField30; }
classBA& Field30(int field30)
{
m_iField30 = field30;
return *this;
}
...
}
=> 缺点:每次创建一个实例都会导致许多函数调用并且非常昂贵。
方法 4> 将映射传递给基类和派生类的所有构造函数。
=> 缺点:我真的认为这是个坏主意,尽管它使数据传递变得容易。
challenge-2> 基类的默认值由其不同的派生类决定。
例如,classA::m_iField2的默认值根据派生类的不同而不同。
针对这个问题,我可以想出两个解决方案,但没有一个对我有吸引力。
方法一> 将默认逻辑添加到派生类本身。
方法二> 将默认逻辑添加到工厂类本身。
我列出了所有我能想到的方法。但是,我仍然在寻找一个干净专业的解决方案来解决这个问题。如果有一个编写良好的 API 库,我可以将其用作解决类似问题的引用,那将是最好的。 欢迎任何评论。
谢谢
/////////////////////// framework ////////////////////////////////////////
// Note:
// <1> the class hierarchy has to kept as this
// <2> getter and setter functions in each class have to kept as this
// <3> add new functions(i.e constructors) are allowed
// <4> add new classes or structures are allowed
/////////////////////////////////////////////////////////////////////////
#include "stdafx.h"
#include <map>
#include <string>
#include <iostream>
using namespace std;
/************************************************************************/
/* Class Name: classA (an abstract base class)
* default value of m_iField2 is determined by its derived class
/************************************************************************/
class classA
{
public:
virtual ~classA() = 0 {}
// ...
private: //
int m_iField1;
float m_iField2; // one of the potential field that has to get the default value
// ...
double m_iField29;
};
/************************************************************************/
/* Class Name: classBA
* If the pass-in parameters do NOT include value for the field classA::m_iField2
* then assign its value as 200.0f
/************************************************************************/
class classBA : public classA
{
// ...
private:
int m_iField30;
// ...
double m_iField40;
};
/************************************************************************/
/* Class Name: classCA
* If the pass-in parameters do NOT include value for the field classA::m_iField2
* then assign its value as 300.0f
/************************************************************************/
class classCA : public classA
{
// ...
private:
int m_iField50;
// ...
int m_iField60;
};
int main(int argc, char* argv[])
{
map<string, string> mapStrsBA;
mapStrsBA["name"] = "classBA";
mapStrsBA["field1"] = "5";
// ...
mapStrsBA["field40"] = "1.89";
// pass mapStrsBA to a factory class with function to create a new instance of class classBA
map<string, string> mapStrsCA;
mapStrsBA["name"] = "classCA";
mapStrsBA["field1"] = "6";
// ...
mapStrsBA["field60"] = "19";
// pass mapStrsCA to a factory class with function to create a new instance of class classCA
return 0;
}
最佳答案
让我在你的挑战 1 中添加方法 5,它并不总是适用,但经常适用,尤其是如果你碰巧有很多字段:找到严格属于一起的成员字段,并收集它们成逻辑对象。例如,假设您有一个如下所示的类 Shape:
class Shape
{
public:
Shape(pass initial values for all member variables);
// ...
private:
// bounding box coordinates
int xmin, xmax;
int ymin, ymax;
// color
int red, green, blue;
int alpha;
// center point (for rotations)
int cx, cy;
};
这是 10 个变量。然而,这些并不是真正无关的。其中大部分是 x/y 对,然后有一组指定颜色。因此你可以重写如下:
struct Point
{
int x, y;
Point(int ax, int ay): x(ax), y(ax) {}
};
struct Color
{
int red, green, blue, alpha;
Color(int r, int g, int b, int a): red(r), green(g), blue(b), alpha(a) {}
};
class Shape
{
public:
// ...
private:
// bounding box
Point lower_left, upper_right;
Color color;
Point center;
};
现在您突然只有四个变量可以传递了。您甚至可能会考虑制作一个由两个角点组成的矩形类型,并将其用于边界框,进一步将变量数量减少到 3。
请注意,不仅减少了要传递的参数数量,还增加了清晰度。
既然你没有提供任何关于你的类的细节,我不能说这样的逻辑分组是否也适用于你的类,但考虑到你提到的大量参数,如果不能的话我会感到惊讶。
对于挑战 2,我认为将默认逻辑添加到派生类会更好。
关于c++ - 创建派生类的模式,派生类本身和基类都包含许多字段,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8585161/
出于纯粹的兴趣,我很好奇如何按顺序创建PI,而不是在过程结果之后生成数字,而是让数字在过程本身生成时显示。如果是这种情况,那么数字可以自行产生,我可以对以前看到的数字实现垃圾收集,从而创建一个无限系列。结果只是在Pi系列之后每秒生成一个数字。这是我通过互联网筛选的结果:这是流行的计算机友好算法,类机器算法:defarccot(x,unity)xpow=unity/xn=1sign=1sum=0loopdoterm=xpow/nbreakifterm==0sum+=sign*(xpow/n)xpow/=x*xn+=2sign=-signendsumenddefcalc_pi(digits
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
当我使用Bundler时,是否需要在我的Gemfile中将其列为依赖项?毕竟,我的代码中有些地方需要它。例如,当我进行Bundler设置时:require"bundler/setup" 最佳答案 没有。您可以尝试,但首先您必须用鞋带将自己抬离地面。 关于ruby-我需要将Bundler本身添加到Gemfile中吗?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/4758609/
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我主要使用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
使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta
为了将Cucumber用于命令行脚本,我按照提供的说明安装了arubagem。它在我的Gemfile中,我可以验证是否安装了正确的版本并且我已经包含了require'aruba/cucumber'在'features/env.rb'中为了确保它能正常工作,我写了以下场景:@announceScenario:Testingcucumber/arubaGivenablankslateThentheoutputfrom"ls-la"shouldcontain"drw"假设事情应该失败。它确实失败了,但失败的原因是错误的:@announceScenario:Testingcucumber/ar
我对最新版本的Rails有疑问。我创建了一个新应用程序(railsnewMyProject),但我没有脚本/生成,只有脚本/rails,当我输入ruby./script/railsgeneratepluginmy_plugin"Couldnotfindgeneratorplugin.".你知道如何生成插件模板吗?没有这个命令可以创建插件吗?PS:我正在使用Rails3.2.1和ruby1.8.7[universal-darwin11.0] 最佳答案 随着Rails3.2.0的发布,插件生成器已经被移除。查看变更日志here.现在
我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server
鉴于我有以下迁移: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