具有继承性的编译器行为
Compiler behavior with inheritance
当我在C++中用完强制转换时,我有一个关于编译器行为的问题。例如,我有一个简单的代码:
class Animal {
public:
Animal() {}
void talk() {
std::cout << "I am an animal" << std::endl;
}
};
class Dog :public Animal {
public:
Dog(){}
void talk() {
std::cout << "I am a dog" << std::endl;
}
void eat() {
std::cout << "eating" << std::endl;
}
};
int main()
{
Animal* animal = new Dog();
animal->talk();//Output is "I am an animal"
//animal->eat();//Compilation error.
return 0;
}
我的问题是,当我运行这个程序时,编译器首先去哪里?它是在Animal类中查找方法,然后因为我没有使用virtual,它调用Animal的talk()方法,还是先检查Dog类中是否存在该方法,然后调用Animal的方法?
给定您的代码,animal->talk()
将始终调用Animal::talk()
。因为animal
是Animal *
并且Animal::talk()
不是虚拟的。
现在,如果您将Animal::talk()
设为虚拟,animal->talk()
将调用Dog::talk()
。通常,这是通过在animal
的vtable上查看运行时来完成的,以查看对象的真实类型,然后调用最合适的talk()
函数。但是,由于上面有Animal *animal = new Dog()
,编译器可以选择通过跳过vtable查找来优化调用,并直接调用Dog::talk()
,因为在编译时该类型是已知的。
相关文章:
- C/C++编译器通常会删除重复的库吗
- 模板-模板参数推导:三个不同的编译器三种不同的行为
- Win32编译器选项和内存分配
- MSVC多行宏编译器错误
- 静态数据成员的问题-修复链接错误会导致编译器错误
- C++,我收到一个无法理解的编译器错误
- 在线编译器中的分段C++没有打印消息
- 如何解决gcc编译器优化导致的centos双编译器设置中的分段错误
- C/C++预处理器是否可以检测一些编译器选项
- 是否有C++编译器选项允许激进地删除所有函数调用,并将参数传递给具有空体的函数
- C++错误C2600:无法定义编译器生成的特殊成员函数(必须首先在类中声明)
- 我需要知道编译器如何在cpp中使用析构函数
- 编译器如何区分std::vector的构造函数
- CLANG 编译器 说:变量"PTR"可能未初始化
- 告诉c++编译器该参数没有别名
- 使用仅使用一次的变量调用的复制构造函数.这可能是通过调用move构造函数进行编译器优化的情况吗
- 为什么所有C++编译器都会崩溃或挂起此代码
- 编译器如何在使用SFINAE的函数和标准函数之间确定两者是否可行
- 我收到同义重复编译器错误。我应该如何修复"类型"X"的参数与类型"X"的参数不兼容?
- 具有继承性的编译器行为