• C++中的对象切割(Object slicing)问题


     在C++中,当我们把派生类对象向上强制转型为基类对象时,会造成对象切割(Object slicing)问题。
     请看下面示例代码

    #include 
    using namespace std;
    
    class CBase {
    public:
    	virtual ~CBase() = default;
    
    	virtual void foo() {
    		printf("print in Class CBase,this address:%p\n", this);
    	}
    };
    
    class CDerive : public CBase {
    public:
    	virtual void foo() override {
    		printf("print in Class CDerive,this address:%p\n", this);
    	}
    };
    
    int main() 
    {
    	CBase base;
    	CDerive derive;
    	CBase* pBase = &derive;
    
    	cout << "\n本次调用的是pBase->CBase::foo()\n\t";
    	pBase->CBase::foo();
    
    	cout << "\n本次调用的是pBase->foo()\n\t";
    	pBase->foo();
    
    	cout << "\n本次调用的是 (static_cast(derive)).foo() \n\t";
    	(static_cast<CBase>(derive)).foo();
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34

     代码运行结果如下:
    在这里插入图片描述
     对于前两次调用成员函数foo的结果,大家应该都清楚,pBase是指向派生类对象的基类指针,而foo成员函数是虚函数;因此 pBase->foo() 调用的是派生类方法;pBase->CBase::foo()由于指定了调用CBase的foo函数,因此调用的是基类方法。另外,foo函数打印出来的this指针的地址,显示前两次调用的this指针是同一个,也就是指向子类对象derive。
     关键是最后一次调用的结果,显示调用的是基类方法,并且对象的地址也不是子类对象derive的地址。它的调用语句是:

    (static_cast(derive)).foo();

     这是为什么呢?这涉及到对象切割(object slicing)。
     在编译时,编译器看到这句是要将派生类对象derive向上强制转为基类对象,因此编译器以所谓的拷贝构造函数把derive中的CBase成分拷贝一份,并把拷贝好的副本作为临时对象放在内存中。在运行期,这个副本(它是CBase对象)调用foo函数时,当然调用的是CBase::foo成员函数了!
     请注意,如果我们把派生类对象的指针向上强制转为基类对象的指针,那么转换后的基类指针仍然指向派生类对象。
     总之,当我们把派生类对象向上强制转型为基类对象时,会经历对象切割(Object slicing),也就是:编译器在内存中创建了一个子类对象的基类成分的副本,这个临时对象与之前的子类对象无关。

  • 相关阅读:
    Allegro自动沿着目标任意形状走线操作指导
    WPF中ElementName与RelativeSource绑定的局限性以及对策
    k8s 实战篇 - 镜像打包部署 - springboot&mysql - 3
    bat批处理---实现输入指定拷贝文件
    Java基于J2EE的流浪动物收容与领养管理系统
    JAVA工程师面试专题-《消息队列》篇
    Linux(多用户下)查看cuda、cudnn版本、查看已经安装的cuda版本相关命令
    window下编译openssl
    ML Design Patterns——Models and Framework
    说说我的实习收获
  • 原文地址:https://blog.csdn.net/mary288267/article/details/133686995