当前位置: 首页 > news >正文

C++基类指针或引用指向或引用派生类对象(实现动态多态四种手段)父类指针访问子类成员变量(需要dynamic_cast)

文章目录

    • 背景
    • 多态示例:父类指针指向子类对象
    • 父类指针指向子类对象,如何通过父类指针访问到子类特定的成员变量
    • 实现动态多态的四种手段:基类的指针或引用指向或引用一个派生类对象(new或不new)

背景

比如有父类Animal和子类Cat,用new创建对象有以下四种方式:

Animal* p = new Cat();
Animal* pp = new Animal();
Cat* q = new Cat();
Cat* qq = new Animal();

其中第四种是不合法的。

第二种和第三种都好理解,但是第一种,用父类指针指向了子类对象,这种是干嘛用的的🧐

这是用来实现多态的

多态示例:父类指针指向子类对象

使用父类指针指向子类对象时,可以实现多态性,即在运行时根据对象的实际类型来调用相应的函数。这样可以方便地实现代码的扩展和维护,同时也可以提高代码的可读性和可维护性。

下面是一个简单的示例代码,演示了使用父类指针指向子类对象的好处:

#include <iostream>
using namespace std;class Animal
{
public:virtual void speak(){cout << "Animal speaks" << endl;}string name;
};class Cat : public Animal
{
public:void speak(){cout << "Cat speaks" << endl;}string name;
};class Dog : public Animal
{
public:void speak(){cout << "Dog speaks" << endl;}
};int main()
{Animal *p1 = new Cat();Animal *p2 = new Dog();p1->speak(); // 输出 Cat speaksp2->speak(); // 输出 Dog speaksdelete p1;delete p2;return 0;
}

在上面的代码中,Animal是一个基类,Cat和Dog是Animal的子类。在main函数中,我们使用Animal指针p1和p2分别指向Cat对象和Dog对象。由于speak函数是虚函数,因此在运行时会根据对象的实际类型来调用相应的函数。因此,p1->speak()会调用Cat类中的speak函数,输出"Cat speaks",p2->speak()会调用Dog类中的speak函数,输出"Dog speaks"。

如果我们不使用父类指针指向子类对象,而是直接使用Cat和Dog对象,那么我们就需要分别调用它们的speak函数,这样会导致代码的重复和冗余,不利于代码的扩展和维护。

父类指针指向子类对象,如何通过父类指针访问到子类特定的成员变量

当使用父类指针指向子类对象时,无法直接访问子类中特定的成员变量。但是,可以通过将父类指针转换为子类指针或引用来访问子类中特定的成员变量。

在C++中,可以使用dynamic_cast运算符将父类指针或引用转换为子类指针或引用,从而访问子类中特定的成员变量。需要注意的是,dynamic_cast运算符只能用于含有虚函数的类类型之间的转换,而且转换时会进行类型检查,如果转换失败会返回空指针或引用。

下面是一个简单的示例代码,演示了如何使用dynamic_cast运算符访问子类中特定的成员变量:

#include <iostream>
using namespace std;class Animal
{
public:virtual void speak(){cout << "Animal speaks" << endl;}
};class Cat : public Animal
{
public:string name;void speak(){cout << "Cat speaks" << endl;}
};int main()
{Animal *p = new Cat();p->speak(); // 输出 Cat speaksCat *q = dynamic_cast<Cat *>(p);if (q != nullptr){q->name = "Tom";cout << q->name << endl; // 输出 Tom}delete p;return 0;
}

在上面的代码中,Animal类中没有定义name成员变量,而Cat类中定义了name成员变量。在main函数中,我们使用Animal指针p指向Cat对象,然后使用dynamic_cast运算符将p转换为Cat指针q,从而访问Cat类中的name成员变量。需要注意的是,在转换时需要进行类型检查,如果转换失败会返回空指针。

参考文章:C++四种cast转换(const_cast、static_cast、dynamic_cast、reinpreter_cast)类型转换运算符

实现动态多态的四种手段:基类的指针或引用指向或引用一个派生类对象(new或不new)

可以定义一个基类的指针或引用,然后将其指向或引用一个派生类的对象。这样,通过基类的指针或引用调用虚函数时,会根据实际指向的对象类型来动态调用相应的函数,从而实现多态。

下面是一个使用指针实现多态的示例代码:

#include <iostream>
using namespace std;// 重写示例
class Base
{
public:virtual void func(){cout << "Base::func()" << endl;}
};class Derived : public Base
{
public:virtual void func(){cout << "Derived::func()" << endl;}
};int main()
{// 方法1:new,父类指针指向子类对象Base *p = new Derived();p->func(); // 输出 Derived::func()delete p;cout << "----" << endl;// 或者// 方法2:不new,父类指针指向子类对象Derived dd;Base *pp = &dd;pp->func(); // 输出 Derived::func()cout << "----" << endl;// 或者// 方法3:不new,父类引用引用子类对象Derived ddd;Base &ppp = ddd;ppp.func(); // 输出 Derived::func()cout << "----" << endl;// 或者// 方法4:new,父类引用引用子类对象Base &pppp = *(new Derived());pppp.func(); // 输出 Derived::func()delete &pppp;return 0;
}

在上面的代码中,定义了一个基类的指针或引用,然后将其指向一个派生类对象。通过基类的指针或引用调用虚函数时,会根据实际指向的对象类型来动态调用相应的函数,从而输出派生类中的func()函数的内容。

http://www.lryc.cn/news/92613.html

相关文章:

  • WTM框架运行报错0308010C:digital envelope routines::unsupported
  • (二)CSharp-索引器
  • 配合AI刷leetcode 实现1170
  • English Learning - L3 作业打卡 Lesson5 Day36 2023.6.9 周五
  • 前端框架笔记
  • 详细设计文档
  • Java011——Java数据类型转换(基本数据类型)
  • mybatis-plus用法(二)
  • SQL笔记-存储过程+循环
  • HNU-操作系统OS-作业1(4-9章)
  • springboot 精华
  • 我用ChatGPT写2023高考语文作文(三):新课标I卷
  • HTML 标签的学习
  • 计算耗时为微秒的方法(包含:时/分/秒/毫秒/微秒/纳秒)
  • 通过 Python 封装关键词搜索阿里巴巴商品api接口
  • 分布式光伏消纳的微电网群共享储能配置策略研究(Matlab代码实现)
  • C语言写网络爬虫总体思路
  • 机器学习实战六步法之训练模型、优化模型、部署模型(七)
  • 《C++高级编程》读书笔记(七:内存管理)
  • Scrum团队的三个角色
  • python环境中使用 requirement.txt 安装依赖
  • UE5 材质常用大全
  • 笔记本安装centos操作系统
  • Polarion工作流插件(自定义)
  • JavaScript库:jQuery,简化编程
  • [springboot]菜鸟学习- JdbcTemplate用法浅尝
  • 11.无监督学习之主成分分析
  • 「HTML和CSS入门指南」figcaption 标签详解
  • 电子企业实施数字化工厂建设之前,需要注意哪些
  • 迅捷pdf实现多页插入