在什么条件下,纯虚拟方法生成了
Under what conditions is the pure virtual method stub generated?
我已经在许多地方(例如在此)阅读了有关人们在 Runtime 上获得"纯虚拟方法"称为"错误和程序崩溃"的人。这个答案说
大多数编译器将此类VTable条目分配给一个存根,该存根在流产程序之前发出错误。
,这个甚至说Itanium abi指定了该存根。
问题是:我所有试图看到这种行为中的行为的尝试都被编译器(GCC 6.4.1)抓住, compile Time 作为未定义的参考。例如,从抽象类的构造函数调用纯虚拟函数时,我会得到一个警告
pure virtual ‘virtual int X::f()’ called from constructor
,但同时根本没有为X::f()
生成任何代码,因此接下来是
undefined reference to 'X::f()'
链接器和汇编失败。这似乎是防止运行时错误的一种万无一失的方式。在哪种情况下,我的编译器实际上需要生成上述存根?还是足够聪明,可以尽早检测所有可能的病理情况?
通常会从构造函数中间接称为。这是一个最小的例子:
#include <iostream>
struct X {
virtual void foo() = 0;
void bar() { foo(); }
X() { bar(); std::cout << "X"; }
};
struct Y : X {
void foo() override {}
};
int main() {
Y y;
return 0;
}
如果呼叫直接出现在c'tor内部(因此,可以为纯虚拟函数产生有用的错误消息)。但是,当呼叫是间接的时,从另一个成员中,必须动态派遣。
自然地,Y
部分尚未在X
的构建过程中构造,因此整个事情在不确定的行为的烈火中崩溃了。
live示例 - 使用存根
相关文章:
- 用常见虚拟函数实现的任意组合来实现派生类的正确方法是什么
- 在模板基类中为继承类中的可选重写生成虚拟方法
- 跨 DLL 边界访问虚拟方法是否安全/可能?
- 是否可以使用基类非虚拟方法中的派生类虚拟方法?
- 如何编写 operator= 用于使用虚拟方法与非平凡成员的匿名联合
- 让编译器告诉什么确切的纯虚拟方法使结构抽象?
- 使用模板而不是虚拟方法的管道模式
- 派生类调用父类的方法,该方法调用重写的虚拟方法调用错误的方法
- 从纯虚拟类 (A) 派生的指针无法访问来自纯类 (B) 的重载方法
- 为什么调用没有正文的纯虚拟方法不会导致链接器错误?
- 出于什么目的,非虚拟方法将与C++一起使用?
- 为什么使用存储在虚拟方法表中的地址调用虚拟函数的函数会返回垃圾?
- 如何重写继承的嵌套类中存在的虚拟方法
- 私有虚拟方法有什么用?
- 派生类中纯虚拟基方法的专业化
- 基类可以声明虚拟方法但不定义它吗?仍然在派生类中定义
- googletest:测试基类具有纯虚拟方法的派生类时的核心转储
- 确保模拟的 GTest 方法覆盖虚拟方法
- 使用回调函数从构造函数调用虚拟/派生方法的替代方法?
- 如何调用孩子的方法:虚拟关键字不起作用