C语言对象编程第三弹:多态

多态的概念及C++例子

关于多态,之前整理的《什么是面向对象?》这篇文章有说到:

多态按字面的意思就是多种形态。当类之间存在层次结构,并且类之间是通过继承关联时,就会用到多态。

多态意味着调用成员函数时,会根据调用函数的对象的类型来执行不同的函数。

比如关于多态的C++的例子(该C++代码来自菜鸟教程):

左右滑动查看全部代码>>>

#include <iostream> 
using namespace std;

// 基类  
class Shape 
{
   protected:
      int width, height;
   public:
      Shape( int a=0, int b=0)
      {
         width = a;
         height = b;
      }
      virtual int area()
      {
         cout << "Parent class area" <<endl;
         return 0;
      }
};

// 派生类Rectangle
class Rectangle: public Shape
{
   public:
      Rectangle( int a=0, int b=0):Shape(a, b) { }
      int area ()
      { 
         cout << "Rectangle class area" <<endl;
         return (width * height); 
      }
};

// 派生类Triangle
class Triangle: public Shape
{
   public:
      Triangle( int a=0, int b=0):Shape(a, b) { }
      int area ()
      { 
         cout << "Triangle class area" <<endl;
         return (width * height / 2); 
      }
};

// 程序的主函数
int main( )
{
   Shape *shape;
   Rectangle rec(10,7);
   Triangle  tri(10,5);
 
   // 存储矩形的地址
   shape = &rec;
   // 调用矩形的求面积函数 area
   shape->area();
 
   // 存储三角形的地址
   shape = &tri;
   // 调用三角形的求面积函数 area
   shape->area();
   
   return 0;
}

编译、运行结果为:

代码中用到了一个关键字:virtual,这是C++的关键字。基类中用virtual关键字修饰的函数叫做虚函数

这虚函数有点像弱定义的感觉,先定义一个弱的/虚的函数,其它地方再定义同名的真的函数,实际用的是真的函数。

该例中,在派生类中重新定义基类中定义的虚函数area时,会告诉编译器不要静态链接到该函数,而是根据所调用的对象类型来选择调用真正的函数。

假如这个例子中不使用virtual来修饰基类中的area函数,则上例输出结果则为:

显然,如果没有virtual来修饰的话,用到的都是基类中的area。

本篇笔记我们还需要知道一个知识:虚函数表。具体介绍如(图片截图自百度百科):

本篇笔记关于C++相关知识的就不再拓展,感兴趣的朋友可自行查资料进行学习。下面来看看C语言中怎么来实现上诉的例子:

C语言多态实例分析

这一节我们用C语言来实现上述例子的功能。下面看具体实现:

1、虚函数表

首先,我们可以使用函数指针来模拟C++的虚函数表:

/* 模拟C++的虚函数表 */
typedef struct _Ops
{
 int (*area)(void);
}Ops;

2、基类Shape:

/* 基类 */  
typedef struct _Shape 
{
 Ops ops;
 int width;
 int height;
}Shape;

3、派生类Rectangle、Triangle

/* 派生类Rectangle */
typedef struct _Rectangle
{
 Shape shape;
 char rectangle_name[20];
}Rectangle;

/* 派生类Triangle */
typedef struct _Triangle
{
 Shape shape;
 char triangle_name[20];
}Triangle;

4、两个派生类对应的area函数

/* Rectangle的area函数 */
int rectangle_area(void)
{
 printf("Rectangle class area\n");
}

/* Triangle的area函数 */
int triangle_area(void)
{
 printf("Triangle class area\n");
}

5、主函数/测试函数

左右滑动查看全部代码>>>

/* 主函数 */
int main(void)
{
 Rectangle rectangle;
 memset(&rectangle, 0, sizeof(Rectangle));
 rectangle.shape.ops.area = rectangle_area; /* 与自己的area函数做绑定 */

Triangle triangle;
 memset(&triangle, 0, sizeof(Triangle));
 triangle.shape.ops.area = triangle_area; /* 与自己的area函数做绑定 */

Shape *shape;

shape = (Shape*)&rectangle;
 shape->ops.area();

shape = (Shape*)&triangle;
 shape->ops.area();
 
 return 0;
}

编译、运行结果为:

与C++例子中得到的结果是一样的。即父类指针shape来操作两个子类时,使用相同的接口时调用了不同的函数:

以上实现了简单的多态的功能。

这个例子中我们的操作函数(虚函数)只有一个,即area函数。

假如有多个操作函数,我们可以再建个结构体变量(函数表)把这些函数再包一层,这样会更清晰些。

在这个例子中,有如下对应关系:

因为这里只有一个操作函数,所以就没有建立一个函数表来包装一层了。我们可以再加一个函数表,如:

有多个函数的话,就更有必要构建一个函数表了:

这种方式是不是很熟悉了?如在《通俗易懂,嵌入式Linux驱动基础》就是这样的套路:

Linux内核给我们提供一个文件操作的结构体模板,我们需要用到什么依次实现、依次填充函数表,这样就很清晰。

除此之外,这里的给结构体初始化的方式使用如下这种方式:

(0)

相关推荐

  • Java多态性:Java什么是多态?

    多态性是面向对象编程的又一个重要特征,它是指在父类中定义的属性和方法被子类继承之后,可以具有不同的数据类型或表现出不同的行为,这使得同一个属性或方法在父类及其各个子类中具有不同的含义. 对面向对象来说 ...

  • Halcon 筛选图中面积最大的Region

    筛选图中面积最大的Region 首先要操作的就是分离出图中的Region,分离方法就是传统的阈值(threshold),连接(connection),将空洞填充(fill_up). 将这些操作完之后, ...

  • 每晚一首英文儿歌NO99——The Shape Song 2

    每晚一首英文儿歌 让孩子感受英文和韵律的美 NO99:The Shape Song 2 ⇑⇑⇑点击上方绿色小喇叭收听 歌 词 一起来唱 The Shape Song 2 Okay everybody. ...

  • C语言对象编程第一弹:封装与抽象

    前言 上次整理了一篇关于面向对象的笔记:<什么是面向对象?>.简单地分享了面向对象的一些基础知识. C语言虽不是面向对象的语言,但也可以使用面向对象的思想来设计我们的程序. C语言 + 面 ...

  • C语言对象编程第二弹:继承

    前言 继承简单说来就是父亲有的东西,孩子可以继承过来. 当创建一个类时,我们不需要重新编写新的数据成员和成员函数,只需指定新建的类继承了一个已有的类的成员即可. 这个已有的类称为基类,新建的类称为派生 ...

  • C语言经典编程(浙大版C语言第三版)详解

    C语言经典编程详解 -- 一览表: 1.[C语言经典编程]练习2-1 Programming in C is fun! (5分) 2.[C语言经典编程]练习2-3 输出倒三角图案 (5分) 3.[C语 ...

  • 国产步枪七十年发展史(第三弹)

    "先进但落后""想法对却也做不到",这样的矛盾学用今天的主角身上却也一点都不突兀,甚至可以说有点契合. 这个"它"也就是本篇的主角" ...

  • 告白第三弹 | 1000份爱的告白,哪一个属于你?

    不平凡的2020已经过去,2021年的第一个月也转眼过半.新一年,意味新开始,带来新希望.对蒲公英女性领导力研究院来说,在201年的元月迎来了七岁生日! 七年,在蒲公英女性领导力研究院的推动下,&qu ...

  • 【心理健康】心理健康知识第三弹之阿伦森效应

                    第三弹 心理健康知识 之阿伦森效应 五一小长假结束后再次即将迎来调休,心理工作站的周例心理科普又来与你见面啦!什么是阿伦森效应?就让心理工作站现在来带你研究吧! 阿伦森 ...

  • 诸葛亮兵法:怎样成为上级领导的重点培养对象?这三点做好就行!

    从基层员工到基层干部,这是职场人士晋升的第一步.然而大家不要小看这小小的进步,因为在现实生活中大部分人混迹职场多年都始终没有迈出这一步.原因何在?是因为大部分人都没有搞清楚,应当怎么做才能使自己成为领 ...

  • 高考语文,句式、内容和语言风格,抓住三个出题角度,搞定改句题

    高考语文,句式、内容和语言风格,抓住三个出题角度,搞定改句题

  • 新闻评论第三弹 | 「夸夸群式评论」需要的辩证思维

    上个星期布置的评论内容是关于阿里平台助力乡村直播带货,说实话这个内容并不是平常的一些社会舆论事件,而是一个当前零售市场中出现的新现象--线上直播带动线下产品销售,尤其特指为目前的乡村特色产品拓展线上行 ...