学懂C++(五十):深入详解 C++ 陷阱:对象切片(Object Slicing)

2024-09-03 04:44

本文主要是介绍学懂C++(五十):深入详解 C++ 陷阱:对象切片(Object Slicing),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

对象切片基本概念

什么是对象切片?

对象切片(Object Slicing)是 C++ 中的一个常见陷阱,发生在将派生类对象赋值给基类对象时。由于基类对象无法存储派生类特有的数据和行为,因此派生类对象的特有部分会被“切掉”,只保留基类部分。

为什么会发生对象切片?

        对象切片发生的原因在于 C++ 的赋值操作是基于值语义的。当你将一个派生类对象赋值给一个基类对象时,实际上是创建了一个新的基类对象,而这个新的基类对象只能包含基类的成员变量和方法。因此,派生类对象的特有部分(即派生类独有的数据和方法)就会被丢弃,这就是对象切片。

示例代码

让我们通过一个具体的例子来深入理解对象切片。

#include <iostream>//基类
class Base {
public:Base(int a): a_(a) {}virtual void show() const {std::cout << "Base class, a: " << a_ << std::endl;}
private:int a_;
};//派生类
class Derived : public Base {
public:Derived(int a, int b): Base(a), b_(b) {}virtual void show() const override {Base::show();std::cout << "Derived class, b: " << b_ << std::endl;}
private:int b_;
};int main() {Derived d(5, 10);Base b = d; // 对象切片,b_ 被“切掉”b.show();   // 只调用 Base 类的 show(),不会显示 b_return 0;
}

 运行结果

Base class, a: 5

如上所示,派生类的 show() 方法没有被调用,派生类对象的成员变量 b_ 也没有被显示。这是因为 bBase 类型的对象,无法存储 Derived 类型的特有部分。对象切片发生在将派生类对象 d 赋值给基类对象 b 时,导致派生类对象的特有数据 b_ 被丢弃。

如何避免对象切片?

有多种方法可以避免对象切片,以下是几种常见的方法。

1. 使用指针或引用

使用指针或引用可以避免对象切片,因为它们不会创建基类对象的副本。这种方法确保派生类的特有部分不会被“切掉”。

int main() {Derived d(5, 10);Base* pb = &d; // 使用指针pb->show();   // 调用 Derived 类的 show() 方法Base& rb = d; // 使用引用rb.show();    // 调用 Derived 类的 show() 方法return 0;
}
运行结果
Base class, a: 5
Derived class, b: 10
Base class, a: 5
Derived class, b: 10

在这段代码中,pbrb 分别是指向 d 的指针和引用。通过它们调用 show() 方法时,会根据虚函数机制动态绑定到 Derived 类的 show() 方法。

2. 使用多态和虚函数

确保基类中的方法是虚函数,这样即使通过基类指针或引用调用方法,也能动态绑定到派生类的实现。注意:虚函数不能避免对象切片,只能保证在使用指针或引用时实现多态行为。

class Base {
public:Base(int a): a_(a) {}virtual void show() const {  // 使用 virtual 关键字std::cout << "Base class, a: " << a_ << std::endl;}virtual ~Base() {} // 确保基类有虚析构函数
private:int a_;
};class Derived : public Base {
public:Derived(int a, int b): Base(a), b_(b) {}virtual void show() const override {Base::show();std::cout << "Derived class, b: " << b_ << std::endl;}
private:int b_;
};int main() {Derived d(5, 10);Base b = d; // 对象切片仍然发生b.show();   // 只调用 Base 类的 show(),不会显示 b_Base* pb = &d; // 使用指针pb->show();   // 调用 Derived 类的 show() 方法Base& rb = d; // 使用引用rb.show();    // 调用 Derived 类的 show() 方法return 0;
}
 运行结果
Base class, a: 5
Base class, a: 5
Derived class, b: 10
Base class, a: 5
Derived class, b: 10

这里,第一行结果来自于对象切片操作,仅调用了 Base 类的 show() 方法,而没有显示派生类的特有数据 b_。后续使用指针或引用来调用 show() 方法时,动态多态性确保了调用 Derived 类的 show() 方法。

3. 使用智能指针

使用智能指针(如 std::unique_ptrstd::shared_ptr)来管理动态分配的对象,可以有效避免对象切片,同时更好地管理内存。

#include <memory>
#include <iostream>class Base {
public:Base(int a) : a_(a) {}virtual void show() const {std::cout << "Base class, a: " << a_ << std::endl;}
private:int a_;
};class Derived : public Base {
public:Derived(int a, int b) : Base(a), b_(b) {}virtual void show() const override {Base::show();std::cout << "Derived class, b: " << b_ << std::endl;}
private:int b_;
};int main() {std::unique_ptr<Base> pb = std::make_unique<Derived>(5, 10); // 使用智能指针管理动态分配的对象pb->show(); // 调用 Derived 类的 show() 方法std::shared_ptr<Base> spb = std::make_shared<Derived>(5, 10); // 使用智能指针spb->show(); // 调用 Derived 类的 show() 方法return 0;
}
运行结果
Base class, a: 5
Derived class, b: 10
Base class, a: 5
Derived class, b: 10

此例中,pbspb 是指向 Derived 类型对象的智能指针,因此调用了 Derived 类的 show() 方法,避免了对象切片。

总结

        对象切片是 C++ 中一个常见而隐蔽的陷阱,发生在将派生类对象按值赋给基类对象时。由于基类对象无法存储派生类的特有部分,这些特有数据和行为将会被丢弃。要避免对象切片,可以使用指针或引用来管理对象,或者使用智能指针来处理动态分配的对象。此外,通过使用虚函数,可以确保基类指针或引用的多态性行为,但这并不能避免对象切片本身。

        理解和避免对象切片是编写高质量 C++ 代码的关键。掌握这些技术和概念不仅可以提高代码的健壮性,还能使程序更具有扩展性和维护性,避免由于对象切片带来的各种潜在问题。

这篇关于学懂C++(五十):深入详解 C++ 陷阱:对象切片(Object Slicing)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1132060

相关文章

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)