C++雾中风景4:多态引出的困惑,对象的拷贝?

简介: C++作为一门面向对象的语言,自然具备了面向对象的三大特征:封装,继承,多态。在学习多态性质的过程中,发现了C++与其他语言很大的区别(坑?)。在C++中的=操作符的使用与C++呈现的内存模型似乎并不是我所习惯的模式,在拷贝与引用两个不同操作之间摇摆,还是很容易写出存在问题的代码,所以也就引出了今天这篇文章,我们来聊聊=操作符背后的故事。

C++作为一门面向对象的语言,自然具备了面向对象的三大特征:封装,继承,多态。在学习多态性质的过程中,发现了C++与其他语言很大的区别(坑?)。在C++中的=操作符的使用与C++呈现的内存模型似乎并不是我所习惯的模式,在拷贝与引用两个不同操作之间摇摆,还是很容易写出存在问题的代码,所以也就引出了今天这篇文章,我们来聊聊=操作符背后的故事。

1.有些奇怪的多态

来,先上代码,我们从两段要表述多态性质的代码来看看,奇怪在什么地方。

class bird {
public:
    virtual void fly() {
        cout << "I can fly." << endl;
    }
};

class penguin:public bird {
public:
    void fly() {
        cout << "I can't fly." << endl;
    }
}; 

上面是两个继承关系的类定义。penguin(企鹅)类继承了bird类。在bird类之中fly()函数是一个virtual函数,它可以被penguin覆盖。我们看看正确的多态代码应该怎么编写:

int main() {
    bird* b1;
    penguin p; 
    
    b1 = &p;
    b1->fly();   //打印出:"I can't fly." 
}

编译器通过指针的内容,而不是它的类型,来判断应该调用的函数。因此,由于 penguin的对象的地址存储在bird指针中,所以会调用对应的fly()函数。
所以每个bird的子类都可以一个函数fly()的独立实现。这就是多态的使用方式。可以有多个不同的子类,都带有同一个名称但具有不同实现的函数。

啊哈,这一些看起来都很完美。但是熟悉JavaPython的程序员应该会和我一样写出类似于下面的代码吧:

int main() {
    penguin p;
    bird b = p; 
    b.fly(); //打印出:"I can fly."    
}

FxxK,这还是不是我熟悉的多态?为什么输出的内容和我想象的不一样。不行,我得再试一试其他方法。

int main() {
    penguin p;
    ((bird)p).fly(); //同样是打印出:"I can fly."    
}

2.出了什么问题呢?

好吧,上面两段代码我想会让很多JavaPython的程序员深感困惑,看起来C++和我们熟悉的语言想去甚远。其实,这就回到我们今天要聊的主题,接下来我们一一来分析上两段代码:

int main() {
    penguin p;
    bird b = p; 
    b.fly(); //打印出:"I can fly."    
}

其实这段代码最核心的点是弄明白bird b = p语句中的=操作符真正代表的含义。

为了解释这个=操作符,我们继续看下面这段代码。

int main() {
    penguin p;
    bird &b = p; 
    b.fly(); //打印出:"I can’t fly."    
}

有木有很神奇,让我们困惑的问题迎刃而解,只不过添加了一个&操作符。
在C++之中,= 操作符代表一个拷贝

  • bird b = p
    代表b是一个bird对象,通过p拷贝,重新生成一个新的bird对象。所以这是一个拷贝操作,拷贝的是一个对象。
  • bird &b = p
    代表b是一个bird对象的引用,通过p的地址拷贝,重新生成一个新的bird对象的引用。所以这也是一个拷贝操作,拷贝的是一个对象引用。所以通过这个引用,动态调用到p对象真正的函数。

好了,解释完上一段代码之后,我们继续看第二段代码。

int main() {
    penguin p;
    ((bird)p).fly(); //同样是打印出:"I can fly."    
}

这里为什么我们强制类型转换之后,还是没法输出我们想要的结果呢?那是因为

除了指针与引用类型,C++编译器在编译阶段通过类型静态确定调用函数的地址。
通过这句话,我们也不难理解上一段代码输出的结果,所以我们要更好的使用多态,一定要使用好指针和引用。

3.其他语言的困惑的解析

  • Java
    全面放弃了指针与对象拷贝的操作,所以Java之中的=全都是拷贝的对象的引用。也就是我们说的的浅拷贝。(对象拷贝是深拷贝,因为生成新的对象,和原对象不使用同样的内存空间).

  • Python
    同Java一般,都是对象引用。唯一不同的是,Python是动态语言,在实现多态的时候,依赖更多是鸭子类型而不是类原生的继承关系了。

  • Golang
    和Python相同,依赖鸭子类型。

目录
相关文章
|
1月前
|
监控 Linux 测试技术
C++零拷贝网络编程实战:从理论到生产环境的性能优化之路
🌟 蒋星熠Jaxonic,技术宇宙中的星际旅人。深耕C++与零拷贝网络编程,从sendfile到DPDK,实战优化服务器性能,毫秒级响应、CPU降60%。分享架构思维,共探代码星辰大海!
|
9月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
5月前
|
存储 人工智能 编译器
c++--多态
上一篇文章已经介绍了c++的继承,那么这篇文章将会介绍多态。看完多态的概念,你一定会感觉脑子雾蒙蒙的,那么我们先以举一个例子,来给这朦胧大致勾勒出一个画面,在此之前,先介绍一个名词虚函数,(要注意与虚拟继承区分)重定义: 重定义(隐藏)只要求函数名相同(但要符合重载的要求,其实两者实际上就是重载);重定义下:在这种情况下,如果通过父类指针或引用调用函数,会调用父类的函数而不是子类。重定义(或称为隐藏)发生的原因是因为函数名相同但参数列表不同,导致编译器无法确定调用哪一个版本的函数。
107 0
|
8月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
8月前
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
8月前
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
8月前
|
存储 编译器 C++
【c++】多态(多态的概念及实现、虚函数重写、纯虚函数和抽象类、虚函数表、多态的实现过程)
本文介绍了面向对象编程中的多态特性,涵盖其概念、实现条件及原理。多态指“一个接口,多种实现”,通过基类指针或引用来调用不同派生类的重写虚函数,实现运行时多态。文中详细解释了虚函数、虚函数表(vtable)、纯虚函数与抽象类的概念,并通过代码示例展示了多态的具体应用。此外,还讨论了动态绑定和静态绑定的区别,帮助读者深入理解多态机制。最后总结了多态在编程中的重要性和应用场景。 文章结构清晰,从基础到深入,适合初学者和有一定基础的开发者学习。如果你觉得内容有帮助,请点赞支持。 ❤❤❤
1060 0
|
9月前
|
安全 编译器 C语言
【C++篇】深度解析类与对象(中)
在上一篇博客中,我们学习了C++类与对象的基础内容。这一次,我们将深入探讨C++类的关键特性,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载、以及取地址运算符的重载。这些内容是理解面向对象编程的关键,也帮助我们更好地掌握C++内存管理的细节和编码的高级技巧。
|
9月前
|
存储 程序员 C语言
【C++篇】深度解析类与对象(上)
在C++中,类和对象是面向对象编程的基础组成部分。通过类,程序员可以对现实世界的实体进行模拟和抽象。类的基本概念包括成员变量、成员函数、访问控制等。本篇博客将介绍C++类与对象的基础知识,为后续学习打下良好的基础。
|
5月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
150 0