C++类和虚拟方法
C++ Class and virtual method
我有两个关于虚拟方法的问题。
第一:
class Parent
{
public:
virtual void SHOW(int x = 5) { cout << "PARENT " << x << endl; }
};
class Child : public Parent
{
public:
virtual void SHOW(int y = 10) { cout << "CHILD " << y << endl; }
};
int main()
{
Child Y;
Parent* P = &Y;
P->SHOW();
getch();
return 0;
}
我认为 tt 应该是孩子 10,但结果是孩子 5
另一个:
class Parent
{
public:
virtual void SHOW() { cout << "PARENT" << endl; }
};
class Child : public Parent
{
private:
virtual void SHOW() { cout << "CHILD" << endl; }
};
int main()
{
Child Y;
Parent* P = &Y;
P->SHOW();
getch();
return 0;
}
它会在屏幕上显示孩子。我不知道从外面调用的私有方法如何?
谢谢。我正在学习英语,所以..:)
1)C++标准说
虚函数调用 (10.3) 在虚函数的声明中使用默认参数,这些参数由表示对象的指针或引用的静态类型确定。派生类中的重写函数不会从它重写的函数中获取默认参数。
§8.3.6/10
2)在C++中用私有函数覆盖公共虚拟函数
虚函数调用使用虚函数声明中的默认参数,这些参数由用于调用函数的指针的静态类型确定。在您的情况下,P 的静态类型是父级,尽管动态类型解析为子级。这是根据C++标准。
这是一个很好的例子,说明为什么通常最好将虚拟函数设为私有,而将公共函数设为非虚拟函数。赫伯·萨特(Herb Sutter)曾经称之为非虚拟接口习语(NVI)。
让我们将其应用于您的代码。我还将函数名称"SHOW"更改为"show",因为C++中的全大写通常表示宏。此外,我将添加一个虚拟析构函数,因为新手正在阅读这些问题,他们不应该看到没有多态类的多态类。请注意,析构函数是 NVI 的例外。
来吧:
class Parent
{
public:
void show(int x = 5) { doShow(x); }
virtual ~Parent() {}
private:
virtual void doShow(int x) { cout << "PARENT " << x << endl; }
};
class Child : public Parent
{
private:
virtual void doShow(int x) { cout << "CHILD " << x << endl; }
};
默认参数通常对于使用类对象的代码很有用。然后,该代码使用公共函数。
Parent *p = new Child;
p->show(); // here's where a default argument is a useful feature
delete p;
然而,继承类对完全不同的东西感兴趣,即基类的虚函数。事实证明,当您为子类编写代码时,您很少会觉得需要默认参数。
底线:
- 默认参数和虚函数不应混合使用。在斯科特·迈耶斯(Scott Meyers)著名的有效C++书中有一个项目:"永远不要重新定义继承的默认参数值"。你应该读那本书。 使用 NVI,
- 您通常不会遇到此问题,因为事实证明默认参数对于公共函数来说更像是一回事,而对于 NVI,公共函数不是虚拟的。
相关文章:
- 用常见虚拟函数实现的任意组合来实现派生类的正确方法是什么
- 在模板基类中为继承类中的可选重写生成虚拟方法
- 跨 DLL 边界访问虚拟方法是否安全/可能?
- 是否可以使用基类非虚拟方法中的派生类虚拟方法?
- 如何编写 operator= 用于使用虚拟方法与非平凡成员的匿名联合
- 让编译器告诉什么确切的纯虚拟方法使结构抽象?
- 使用模板而不是虚拟方法的管道模式
- 派生类调用父类的方法,该方法调用重写的虚拟方法调用错误的方法
- 从纯虚拟类 (A) 派生的指针无法访问来自纯类 (B) 的重载方法
- 为什么调用没有正文的纯虚拟方法不会导致链接器错误?
- 出于什么目的,非虚拟方法将与C++一起使用?
- 为什么使用存储在虚拟方法表中的地址调用虚拟函数的函数会返回垃圾?
- 如何重写继承的嵌套类中存在的虚拟方法
- 私有虚拟方法有什么用?
- 派生类中纯虚拟基方法的专业化
- 基类可以声明虚拟方法但不定义它吗?仍然在派生类中定义
- googletest:测试基类具有纯虚拟方法的派生类时的核心转储
- 确保模拟的 GTest 方法覆盖虚拟方法
- 使用回调函数从构造函数调用虚拟/派生方法的替代方法?
- 如何调用孩子的方法:虚拟关键字不起作用