9. C++虚函数与多态

简介: 9. C++虚函数与多态

虚函数

virtual修饰的成员函数就是虚函数

  • 虚函数对类的内存影响:需要增加一个指针类型的内存大小
  • 无论多少虚函数,只会增加一个指针类型的内存大小
  • 虚函数表的概念: 指向虚函数的指针

  • 我们自己也可以通过虚函数表指针去访问函数(一般做这样的操作不写数据类型)
#include <iostream>
#include <string>
using namespace std;
class MM 
{
public:
  virtual void print()        //1.会写
  {
    cout << "第一个虚函数" << endl;
  }
  virtual void printData() 
  {
    cout << "第二个虚函数" << endl;
  }
protected:
};
int main() 
{
  cout << sizeof(MM) << endl;     //2.对类内存影响
  MM mm;
  mm.print();
  mm.printData();
  //了解一下.32位没问题,64位 vs2022 问题
  int** pObject = (int **)(&mm);
  typedef void(*PF)();
  PF pf = (PF)pObject[0][0];
  pf();           //调用第一个虚函数
  pf = (PF)pObject[0][1];
  pf();           //调用第二个虚函数
  return 0;
}

纯虚函数

具有一个或者多个纯虚函数的类型称之为抽象类,抽象类特性:

  • 抽象类不能创建对象
  • 抽象类可以创建对象指针

纯虚函数也是一个虚函数,所以也需要virtual修饰,纯虚函数是没有函数体,函数=0;

#include <iostream>
using namespace std;
//抽象类
class MM 
{
public:
  //纯虚函数
  virtual void print() = 0;
protected:
  string name;
};
int main() 
{
  //MM object;   抽象类不能构建对象
  MM* pMM = nullptr;
  return 0;
}

虚析构函数

virtual修饰的析构函数 就是虚析构函数

  • 当父类指针被子类对象初始化的时候需要用虚析构函数
  • 所有析构函数底层解析其实函数名相同
#include <iostream>
#include <string>
using namespace std;
class MM 
{
public:
  void print() 
  {
    cout << "MM::print" << endl;
  }
  virtual ~MM()         //虚析构函数
  {
    cout << "~MM" << endl;
  }
};
class Son :public MM 
{
public:
  void print() 
  {
    cout << "Son::print" << endl;
  }
  ~Son() 
  {
    cout << "~Son" << endl;
  }
};
int main() 
{
  MM* pMM = new Son;   //构造子类对象,必须构造父类对象在构造自身
  pMM->print();    //MM看类型
  delete pMM;
  pMM = nullptr;
  return 0;
}

虚函数和多态

多态的概念并不重要,重要的是需要知道那个对象指针在特定情况调用那个成员才是重要

多态概念:  指在继承中指针的同一行为的不同结果,举个栗子(男生和女生上厕所,都是上厕所的行为,男生站着,女生蹲着)

实现多态的两个前提条件:

  • 必须是public继承
  • 必须父类存在virtual类型的成员函数,并且子类中存在该函数的同名函数
  • 一定存在指针的引用
#include <iostream>
#include <string>
using namespace std;
class MM
{
public:
  void print()
  {
    cout << "MM::print" << endl;
  }
  virtual void printData() 
  {
    cout << "MM virtual printData" << endl;
  }
  virtual ~MM()         //虚析构函数
  {
    cout << "~MM" << endl;
  }
};
class Son :public MM
{
public:
  void print()
  {
    cout << "Son::print" << endl;
  }
  void printData() 
  {
    cout << "Son printData" << endl;
  }
  ~Son()
  {
    cout << "~Son" << endl;
  }
};
int main() 
{
  //正常对象的访问,不存在多态
  //都是就近原则
  cout << "正常对象访问" << endl;
  MM mmobject;
  mmobject.print();
  mmobject.printData();
  Son sonobject;
  sonobject.print();
  sonobject.printData();
  //正常的指针访问
  cout << "正常指针访问" << endl;
  MM* pMM = new MM;
  pMM->print();
  pMM->printData();
  Son* pSon = new Son;
  pSon->print();
  pSon->printData();
  //非正常的初始化
  //父类指针被子类初始化
  cout << "不正常的指针赋值" << endl;
  MM* pObject = new Son;
  pObject->print();       //没有virutal 看指针类型 调用MM::print
  pObject->printData();     //有virtual 看对象 调用Son::printData
  pObject = new MM;
  pObject->printData();     //调用MM中
  
  cout << "引用类型" << endl;
  MM& girl = sonobject;
  girl.print();
  girl.printData();
  return 0;
}

虚函数在继承特殊现象

#include <iostream>
#include <string>
using namespace std;
class A
{
public:
  virtual void print()
  {
    cout << "A" << endl;
  }
  virtual void printData() final   //禁止子类重写方法
  {
    cout << "A" << endl;
  }
};
//final: 父类中用来禁止子类重写同名方法
//override: 强制重写,起说明作用,表示当前子类当前方法是重写父类
class B :public A 
{
public:
  //重写:子类实现父类虚函数的同名函数
  void print() override
  {
    cout << "B" << endl;
  }
  //void printData(){}  //final禁止重写
};
class C :public B 
{
public:
  void print() 
  {
    cout << "C" << endl;
  }
};
int main() 
{
  B* pb = new C;
  pb->print();      //调用C::print
  pb = new B;
  pb->print();      //调用B::print
  return 0;
}

纯虚函数和ADT

ADT: 抽象数据类型

抽象类本身不能创建对象,但是子类如果重写类父类中纯虚函数,子类是可以被允许创建对象

抽象类一般用于架构项目,构建好整个项目模块,具体细致工作可以交给子类去实现

采用ADT方式设计项目,可以把这个模块构建出来,并且测试代码也可以提前完成。

#include <iostream>
using namespace std;
//抽象产品类
class AbstractProduct 
{
public:
  virtual void printProduct() = 0;
};
//抽象系统类---ADT
//析构函数一定写虚析构函数
class AbstractSystem 
{
public:
  ~AbstractSystem() {}
  virtual void insertData(AbstractProduct* product) = 0;
  virtual void printData()const = 0;
  virtual int size() const = 0;
  virtual bool empty() const = 0;
};
class ArraySystem :public AbstractSystem 
{
public:
  void insertData(AbstractProduct* product) 
  {
  }
  void printData()const 
  {
  }
  int size() const 
  {
    return 0;
  }
  bool empty() const 
  {
    return 0;
  }
};
class ListSystem :public AbstractSystem
{
public:
  void insertData(AbstractProduct* product)
  {
  }
  void printData()const
  {
  }
  int size() const
  {
    return 0;
  }
  bool empty() const
  {
    return 0;
  }
};
int main() 
{
  AbstractSystem* p =new  ArraySystem;
  p->printData();
  //UI中用的比较多
  //MFC --->不需要自己创建,只需要重写一个,构建对象即可
  p = new ListSystem;
  p->printData();
  return 0;
}
目录
相关文章
|
9天前
|
存储 人工智能 编译器
c++--多态
上一篇文章已经介绍了c++的继承,那么这篇文章将会介绍多态。看完多态的概念,你一定会感觉脑子雾蒙蒙的,那么我们先以举一个例子,来给这朦胧大致勾勒出一个画面,在此之前,先介绍一个名词虚函数,(要注意与虚拟继承区分)重定义: 重定义(隐藏)只要求函数名相同(但要符合重载的要求,其实两者实际上就是重载);重定义下:在这种情况下,如果通过父类指针或引用调用函数,会调用父类的函数而不是子类。重定义(或称为隐藏)发生的原因是因为函数名相同但参数列表不同,导致编译器无法确定调用哪一个版本的函数。
|
4月前
|
编译器 C++
c++中的多态
c++中的多态
|
3月前
|
存储 编译器 C++
【c++】多态(多态的概念及实现、虚函数重写、纯虚函数和抽象类、虚函数表、多态的实现过程)
本文介绍了面向对象编程中的多态特性,涵盖其概念、实现条件及原理。多态指“一个接口,多种实现”,通过基类指针或引用来调用不同派生类的重写虚函数,实现运行时多态。文中详细解释了虚函数、虚函数表(vtable)、纯虚函数与抽象类的概念,并通过代码示例展示了多态的具体应用。此外,还讨论了动态绑定和静态绑定的区别,帮助读者深入理解多态机制。最后总结了多态在编程中的重要性和应用场景。 文章结构清晰,从基础到深入,适合初学者和有一定基础的开发者学习。如果你觉得内容有帮助,请点赞支持。 ❤❤❤
386 0
|
5月前
|
Serverless 编译器 C++
【C++面向对象——类的多态性与虚函数】计算图像面积(头歌实践教学平台习题)【合集】
本任务要求设计一个矩形类、圆形类和图形基类,计算并输出相应图形面积。相关知识点包括纯虚函数和抽象类的使用。 **目录:** - 任务描述 - 相关知识 - 纯虚函数 - 特点 - 使用场景 - 作用 - 注意事项 - 相关概念对比 - 抽象类的使用 - 定义与概念 - 使用场景 - 编程要求 - 测试说明 - 通关代码 - 测试结果 **任务概述:** 1. **图形基类(Shape)**:包含纯虚函数 `void PrintArea()`。 2. **矩形类(Rectangle)**:继承 Shape 类,重写 `Print
105 4
|
5月前
|
设计模式 IDE 编译器
【C++面向对象——类的多态性与虚函数】编写教学游戏:认识动物(头歌实践教学平台习题)【合集】
本项目旨在通过C++编程实现一个教学游戏,帮助小朋友认识动物。程序设计了一个动物园场景,包含Dog、Bird和Frog三种动物。每个动物都有move和shout行为,用于展示其特征。游戏随机挑选10个动物,前5个供学习,后5个用于测试。使用虚函数和多态实现不同动物的行为,确保代码灵活扩展。此外,通过typeid获取对象类型,并利用strstr辅助判断类型。相关头文件如&lt;string&gt;、&lt;cstdlib&gt;等确保程序正常运行。最终,根据小朋友的回答计算得分,提供互动学习体验。 - **任务描述**:编写教学游戏,随机挑选10个动物进行展示与测试。 - **类设计**:基类
107 3
|
7月前
|
存储 编译器 数据安全/隐私保护
【C++】多态
多态是面向对象编程中的重要特性,允许通过基类引用调用派生类的具体方法,实现代码的灵活性和扩展性。其核心机制包括虚函数、动态绑定及继承。通过声明虚函数并让派生类重写这些函数,可以在运行时决定具体调用哪个版本的方法。此外,多态还涉及虚函数表(vtable)的使用,其中存储了虚函数的指针,确保调用正确的实现。为了防止资源泄露,基类的析构函数应声明为虚函数。多态的底层实现涉及对象内部的虚函数表指针,指向特定于类的虚函数表,支持动态方法解析。
75 1
|
8月前
|
编译器 C++
C++入门12——详解多态1
C++入门12——详解多态1
100 2
C++入门12——详解多态1
|
8月前
|
C++
C++入门13——详解多态2
C++入门13——详解多态2
139 1
|
10月前
|
编译器 C++ 索引
C++虚拟成员-虚函数
C++虚拟成员-虚函数
|
10月前
|
存储 编译器 C++