虚拟方法和静态/动态定位
Virtual methods and static/dynamic alocation
我想知道为什么在这个例子中静态对象调用父方法而动态对象调用子方法。
#include <string>
#include <iostream>
using namespace std;
class Father {
public:
virtual string Info() {
return "I am father";
}
};
class Son : public Father {
public:
string Info() {
return "I am son";
}
};
int main() {
Father f = Son();
cout << f.Info(); // I am father
Father* pf = new Son();
cout << pf->Info(); // I am son
return 0;
}
这是由于切片-您的f
实际上是一个自动定位的Father
,分配Father f = Son()
实际上是Father f = Father(Son())
,因此f
的动态类型实际上是Father
,并且正如预期的那样,Father::Info()
将被调用。
关于切片的更多信息可以在这篇文章中找到
因为Father f = Son()
创建了Father类型的对象,并从对象Son复制属性(在这种情况下没有要复制的属性),所以Father * pf = new Son()
接收到指向对象Son的指针,它实际上可能是任何Father派生的对象。
最后,在您的示例中,您有3个对象。复制到Father f
的Father f
、unnamed object Son()
,以及指向Son类型对象的Father * pf
。
Son son();
在编译时,您知道son
的类型是Son
。如果动态分配,则可以选择将Son
对象存储在Son
指针或Father
指针中。如果将它们存储在Son
指针中,则与上面相同。但是,如果将Son
对象存储在Father
类型指针中:
Father* ptr = new Son()
那么编译器无法知道Father
指针指向的是哪种类型的对象。因此在运行时,当它发现时,它会调用合适版本的Info
。
相关文章:
- std::向量与传递值的动态数组
- 在c++中用vector填充一个简单的动态数组
- C++中的动态铸造故障
- 如果C++类在类方法中具有动态分配,但没有构造函数/析构函数或任何非静态成员,那么它仍然是POD类型吗
- 内联映射初始化的动态atexit析构函数崩溃
- 使用动态分配的数组会导致代码分析发出虚假的C6386缓冲区溢出警告
- 控制允许动态运行c++的并发操作数
- 如何将这个C++哈希表转换为动态扩展和收缩,而不是使用硬设置的最大值
- 在调用FreeLibrary后,释放动态链接到具有相同版本的CRT堆的DLL的内存
- 输出没有重复元素的动态数组(收缩数组)C++
- C++为线程工作动态地分割例程
- 正在插入动态数组
- 在c++中使用动态分配的问题
- 干净地破坏动态定位对象的向量
- 包含动态分配内存作为值的映射的取消定位速度有多快?
- 动态执行的 ARM ASM 系统调用,可从C++重新定位
- 在C++Builder中动态定位一组图像
- 虚拟方法和静态/动态定位
- c++对象的动态地址重定位
- 未定义的引用,但(动态)库被链接.(可能是坏的重定位地址错误)