- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
尽管使用组合而不是继承?
如果是,有没有语言层面的解决方案?
最佳答案
正如 VonC 所写,但我想指出一些事情。
fragile base class问题通常归咎于 virtual methods (方法的动态分配 – 这意味着如果方法可以被覆盖,则在这种被覆盖的方法的情况下必须调用的实际实现只能在运行时决定)。
为什么这是个问题?你有一个类,你向它添加了一些方法,如果 MethodA()
调用 MethodB()
,你不能保证 MethodB( )
您编写的将被调用,而不是覆盖您的 MethodB()
的子类的其他一些方法。
在 Go 中有 embedding , 但没有 polymorphism .如果将类型嵌入到结构中,则嵌入类型的所有方法都会提升 并将位于 method set 中包装器结构类型。但是您不能“覆盖”提升的方法。当然,您可以添加自己的同名方法,并在包装器结构上调用该名称的方法将调用您的方法,但如果从嵌入式类型调用此方法,则不会将其分派(dispatch)给您的方法,它仍将调用为嵌入类型定义的“原始”方法。
因此,我认为脆弱的基类问题仅以相当缓和的形式存在于 Go 中。
让我们看一个例子。首先在 Java 中,因为 Java“饱受”这种问题的困扰。让我们创建一个简单的 Counter
类和一个 MyCounter
子类:
class Counter {
int value;
void inc() {
value++;
}
void incBy(int n) {
value += n;
}
}
class MyCounter extends Counter {
void inc() {
incBy(1);
}
}
实例化和使用MyCounter
:
MyCounter m = new MyCounter();
m.inc();
System.out.println(m.value);
m.incBy(2);
System.out.println(m.value);
输出符合预期:
1
3
到目前为止一切顺利。现在,如果基类 Counter.incBy()
将更改为:
void incBy(int n) {
for (; n > 0; n--) {
inc();
}
}
基类 Counter
仍然完美无缺且可操作。但是 MyCounter
出现故障:MyCounter.inc()
调用 Counter.incBy()
,后者调用 inc()
但由于动态调度,它会调用 MyCounter.inc()
... 是的... 无限循环。堆栈溢出错误。
现在让我们看看同一个例子,这次是用 Go 编写的:
type Counter struct {
value int
}
func (c *Counter) Inc() {
c.value++
}
func (c *Counter) IncBy(n int) {
c.value += n
}
type MyCounter struct {
Counter
}
func (m *MyCounter) Inc() {
m.IncBy(1)
}
测试它:
m := &MyCounter{}
m.Inc()
fmt.Println(m.value)
m.IncBy(2)
fmt.Println(m.value)
输出符合预期(在 Go Playground 上尝试):
1
3
现在让我们按照在 Java 示例中所做的相同方式更改 Counter.Inc()
:
func (c *Counter) IncBy(n int) {
for ; n > 0; n-- {
c.Inc()
}
}
它运行完美,输出是一样的。在 Go Playground 上试用.
这里发生的是 MyCounter.Inc()
将调用 Counter.IncBy()
,后者将调用 Inc()
,但这Inc()
将是 Counter.Inc()
,因此这里没有死循环。 Counter
甚至不知道 MyCounter
,它没有任何对嵌入器 MyCounter
值的引用。
关于oop - Go 中是否存在脆弱的基类问题?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41819033/
关闭。这个问题是opinion-based .它目前不接受答案。 想改善这个问题吗?更新问题,以便可以通过 editing this post 用事实和引文回答问题. 4年前关闭。 Improve t
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引起辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the he
在面向对象的编程中,“基类”是派生其他类的类(http://en.wikipedia.org/wiki/Base_class)。 但是,基类的反面是什么?换句话说,什么是没有任何子类的类? 编辑:我正
关闭。这个问题需要更多focused .它目前不接受答案。 想改善这个问题吗?更新问题,使其仅关注一个问题 editing this post . 8年前关闭。 Improve this questi
据我了解,OOP 是大型项目最常用的范式。我也知道大系统的一些较小的子集使用其他范式(例如 SQL,它是声明性的),并且我也意识到在较低级别的计算 OOP 并不真正可行。但在我看来,通常更高级别的解决
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用资料或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引起辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the he
最近听说OOP(Java)有9条规则。我只知道四种:抽象、多态、继承和封装。 OOP 还有更多规则吗? 最佳答案 看来您正在寻找的是 Principles of Object-Oriented Des
我曾经在一次采访中被问到“OOP 的 3 个主要概念是什么?”。我回答说,我认为有4个,如下: 继承 封装 抽象 多态性 我说得对吗? 最佳答案 语言要成为面向对象有3个要求: 仅支持封装(对象)的语
我有一个关于特定 OOP 问题的组织的简单问题。 假设我有一个地形类,里面充满了瓷砖。 Tile 类有多个派生类,即 Door。 Door 类有一个名为 open() 的方法,用于打开门,还有一个名为
就目前情况而言,这个问题不太适合我们的问答形式。我们希望答案得到事实、引用资料或专业知识的支持,但这个问题可能会引发辩论、争论、民意调查或扩展讨论。如果您觉得这个问题可以改进并可能重新开放,visit
我是 Go 的新手,然后我通过示例搜索了很多如何拥有带有静态函数/变量的静态类,例如 C#。但是,我找不到任何可以很好地回答它的东西。也许这个问题看起来很愚蠢,但我不喜欢不确定或不完全理解某事。 假设
我曾尝试搜索此问题的答案,但很难用语言表达,而且许多问题要么是关于如何创建类,要么是关于如何做非常具体的事情。我需要更多的实用概述 - 我是自学成才的,我了解对象是什么(以及如何创建它们),但我从未见
在开始编码之前,我通常会尝试在没有太多分析(没有图表)的情况下进行 TDD。通常我发现自己将一个类拆分为其他类以分离关注点。我想知道更深入的分析是否会阻止这种情况。我认为大部分面向对象分析无法预测其中
在阅读单例时,我发现这个解释是使用单例的原因: since these object methods are not changing the internal class state, we can
如这里所述 https://standardofnorms.wordpress.com/2012/09/02/4-pillars-of-object-oriented-programming/ 并作为
我是这个网站的新手,所以如果我在发布问题时做错了什么,请告诉我,以便我下次修复。 我很好奇从单个基类继承多个类是否是糟糕的 OOP 实践。这可能不太合理,所以我要详细说明一下。 例如,假设您正在设计一
我对“工厂相关”设计模式及其 OOP 实现的理解一直很简单。 一个 《工厂法》是类内部的一个方法,它有一个接口(interface)(或抽象类)作为返回类型,并根据一些内部逻辑构造实现该接口(inte
C# 中的“密封”关键字,Java 中的 Final。 因为我几乎从不创建任何图表并且我只使用已经完成的类(来自框架)我多年后仍然不知道为什么有人会“锁定”一个类所以它永远不会被扩展/继承。 它是有益
我正在研究面向对象的概念,抽象概念基本上被描述为对用户隐藏实现。因此,如果类中有一个成员函数并且我们为某些任务调用该函数,抽象表示用户不应该关心事情是如何完成的,而应该只知道正在完成什么。但即使在非面
我是一名优秀的程序员,十分优秀!