背景:
参见 this question in the C++ FAQ对于我需要解决的类似情况,但使用命名构造函数。
我有一个基类,B 类。
我有一个来自 B 的派生类,D 类,它通过函数、成员和额外的内存分配添加了额外的功能。
class B 通过不执行任何操作或从特定于 class D< 的虚函数返回默认值和="">.nullptrs 以多态方式支持附加功能
B 类 使用public static Factory Methods 来构造所有 protected 构造函数。 (参见:Named Constructor Idiom)
D 类 使用public static Factory Methods 构造所有protected constructors,这些构造函数的名称与 B 类不同,并且在 B 类中不可用.
一段时间后,创建了一个新的接口(interface)类,class A。此类有一个接口(interface),这样来自 class A 的派生类必须有一个 getter 函数和一个 setter 函数,它们都需要一个 pointer to a class B 但动态值可以是 class B 或 class D
问题:
我想派生 class A 并为 class B 创建复制构造函数、赋值运算符和/或 setter,但是因为 class A 仅将其成员公开为 B 类型的对象 我无法确定返回的对象是 class B 还是 class D.
我如何仅使用公共(public)接口(interface)正确实现上述内容而不会导致切片或内存问题(包括如果上述设置错误且需要更改)?
可能的解决方案?
我很想尝试几个选项:
1)在B类和声明对象类型的所有派生类型中创建一个成员:
if(getB()->GetType() == "D") {
//Call D::CreateD(...)
} else if(getB()->GetType() == "B") {
//Call B::CreateB(...)
}
2) 动态转换为派生类型并检查是否失败:
if(dynamic_cast<D*>(getB()) == nullptr) {
//Call B::CreateB(...)
} else {
//Call D::CreateD(...)
}
3) 使用一个特定于 class D 的虚拟方法,我知道当在 class B 对象上使用时返回 nullptr:
if(getB()->VirtualMethodSpecificToClassD() == nullptr) {
//Call B::CreateB(...)
} else {
//Call D::CreateD(...)
}
这三种情况都有代码味道:
最佳答案
根据 zneak 的评论,我认为如果您使用工厂方法和私有(private)构造函数,那么拥有一个
virtual B* copy() const { return new B(*this); /* calls private B::B(const B&) */ }
类 B 中的方法,在类 D 中重写(返回一个新的 D* -- 这种协变返回类型的使用是明确允许的在 C++ 中)。
然后你的 A 复制构造函数可以做类似的事情
A::A(const A& other) : b(other.getB()->copy()) {}
它应该可以正常工作。
另一方面,如果您更愿意采用您建议的解决方案之一,我认为第一个是最不刺激的——尽管我更愿意使用枚举而不是字符串,所以您可以使用简单的 switch 语句而不是字符串比较。我相信 LLVM 使用类似这样的东西进行“动态转换”,以避免 C++ RTTI 开销。
关于C++:从基类型指针确定派生类型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19782588/
我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server
我可以得到Infinity和NaNn=9.0/0#=>Infinityn.class#=>Floatm=0/0.0#=>NaNm.class#=>Float但是当我想直接访问Infinity或NaN时:Infinity#=>uninitializedconstantInfinity(NameError)NaN#=>uninitializedconstantNaN(NameError)什么是Infinity和NaN?它们是对象、关键字还是其他东西? 最佳答案 您看到打印为Infinity和NaN的只是Float类的两个特殊实例的字符串
我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)
我正在编写一个方法,它将在一个类中定义一个实例方法;类似于attr_accessor:classFoocustom_method(:foo)end我通过将custom_method函数添加到Module模块并使用define_method定义方法来实现它,效果很好。但我无法弄清楚如何考虑类(class)的可见性属性。例如,在下面的类中classFoocustom_method(:foo)privatecustom_method(:bar)end第一个生成的方法(foo)必须是公共(public)的,第二个(bar)必须是私有(private)的。我怎么做?或者,如何找到调用我的cust
有时我需要处理键/值数据。我不喜欢使用数组,因为它们在大小上没有限制(很容易不小心添加超过2个项目,而且您最终需要稍后验证大小)。此外,0和1的索引变成了魔数(MagicNumber),并且在传达含义方面做得很差(“当我说0时,我的意思是head...”)。散列也不合适,因为可能会不小心添加额外的条目。我写了下面的类来解决这个问题:classPairattr_accessor:head,:taildefinitialize(h,t)@head,@tail=h,tendend它工作得很好并且解决了问题,但我很想知道:Ruby标准库是否已经带有这样一个类? 最佳
几个月前,我读了一篇关于rubygem的博客文章,它可以通过阅读代码本身来确定编程语言。对于我的生活,我不记得博客或gem的名称。谷歌搜索“ruby编程语言猜测”及其变体也无济于事。有人碰巧知道相关gem的名称吗? 最佳答案 是这个吗:http://github.com/chrislo/sourceclassifier/tree/master 关于ruby-寻找通过阅读代码确定编程语言的rubygem?,我们在StackOverflow上找到一个类似的问题:
我正在尝试解析一个CSV文件并使用SQL命令自动为其创建一个表。CSV中的第一行给出了列标题。但我需要推断每个列的类型。Ruby中是否有任何函数可以找到每个字段中内容的类型。例如,CSV行:"12012","Test","1233.22","12:21:22","10/10/2009"应该产生像这样的类型['integer','string','float','time','date']谢谢! 最佳答案 require'time'defto_something(str)if(num=Integer(str)rescueFloat(s
我正在玩HTML5视频并且在ERB中有以下片段:mp4视频从在我的开发环境中运行的服务器很好地流式传输到chrome。然而firefox显示带有海报图像的视频播放器,但带有一个大X。问题似乎是mongrel不确定ogv扩展的mime类型,并且只返回text/plain,如curl所示:$curl-Ihttp://0.0.0.0:3000/pr6.ogvHTTP/1.1200OKConnection:closeDate:Mon,19Apr201012:33:50GMTLast-Modified:Sun,18Apr201012:46:07GMTContent-Type:text/plain
如何将send与+=一起使用?a=20;a.send"+=",10undefinedmethod`+='for20:Fixnuma=20;a+=10=>30 最佳答案 恐怕你不能。+=不是方法,而是语法糖。参见http://www.ruby-doc.org/docs/ProgrammingRuby/html/tut_expressions.html它说Incommonwithmanyotherlanguages,Rubyhasasyntacticshortcut:a=a+2maybewrittenasa+=2.你能做的最好的事情是:
我对如何计算通过{%assignvar=0%}赋值的变量加一完全感到困惑。这应该是最简单的任务。到目前为止,这是我尝试过的:{%assignamount=0%}{%forvariantinproduct.variants%}{%assignamount=amount+1%}{%endfor%}Amount:{{amount}}结果总是0。也许我忽略了一些明显的东西。也许有更好的方法。我想要存档的只是获取运行的迭代次数。 最佳答案 因为{{incrementamount}}将输出您的变量值并且不会影响{%assign%}定义的变量,我