• 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),也就是:编译器在内存中创建了一个子类对象的基类成分的副本,这个临时对象与之前的子类对象无关。

  • 相关阅读:
    tomcat动静分离和负载均衡
    .Net Core中无处不在的Async/Await是如何提升性能的?
    Web3 的通行证——DID 带来数字身份革命
    【Hadoop】Hadoop 高频面试题英语版(1)
    电源自动测试系统-DC/DC 电源自动测试系统NSAT-8000,免费看方案视频效果
    [附源码]计算机毕业设计JAVA教师业绩考核系统
    JVM面试题:(四)四种引用方式强弱软虚
    Debian常用命令
    ISO16000-9建筑产品和家具中挥发性有机物的测试
    安装 ADB 工具步骤以及基本使用
  • 原文地址:https://blog.csdn.net/mary288267/article/details/133686995