当前位置:首页 > 嵌入式 > 嵌入式大杂烩
[导读]点击上方「嵌入式大杂烩」,选择「置顶公众号」第一时间查看编程笔记! 多态的概念及C++例子 关于多态,之前整理的《什么是面向对象?》这篇文章有说到: 多态按字面的意思就是多种形态。当类之间存在层次结构,并且类之间是通过继承关联时,就会用到多态。

点击上方「嵌入式大杂烩」,选择「置顶公众号」第一时间查看编程笔记!

多态的概念及C++例子

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

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

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

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

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

#include <iostream> 
using namespace std;

// 基类  
class Shape 
{

   protected:
      int width, height;
   public:
      Shape( int a=0int 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=0int b=0):Shape(a, b) { }
      int area ()
      

         cout << "Rectangle class area" <<endl;
         return (width * height); 
      }
};

// 派生类Triangle
class Triangle: public Shape
{
   public:
      Triangle( int a=0int 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, 0sizeof(Rectangle));
 rectangle.shape.ops.area = rectangle_area; /* 与自己的area函数做绑定 */

 Triangle triangle;
 memset(&triangle, 0sizeof(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内核给我们提供一个文件操作的结构体模板,我们需要用到什么依次实现、依次填充函数表,这样就很清晰。

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


可能有些朋友没用过这种初始化结构体的方式。这里就顺便提一下,这时使用指定初始化器(designated initializer)的方式,关于指定初始化器的介绍可查阅往期笔记:《C语言中,什么是指定初始化器?》《结构体详解》

以上就是本次的分享,感谢阅读与分享!如有错误,欢迎指出!

猜你喜欢

OpenBLT Bootloader的使用分享

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

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


为了便于公众号读者交流学习,小编创建了相关相关交流群。坑位有限,感兴趣的朋友可以扫码下方二维码加我微信,由我邀请入群:


欢迎大家进群交流、共同进步。同时,我也会关注一些大家问的一些问题,从中挑选一些具有代表性的、并且在我知识范围内的问题写出相关文章做分享。



免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除( 邮箱:macysun@21ic.com )。
换一批
延伸阅读

在C语言开发中,位操作符是最容易被新手忽略,却能在嵌入式开发、底层驱动、算法优化中发挥巨大作用的工具。和常规的算术操作、逻辑操作相比,位操作直接操作二进制位,执行效率更高,占用代码空间更小,能轻松实现很多用常规方法很难实...

关键字: C语言 位操作符

我们初学编程时,总默认浮点数就是小数的代名词,好像二者天生就是绑定在一起的:整数用整型存,小数就用浮点数存,这似乎是天经地义的规则。但如果我们仔细观察,总会遇到一些难以理解的奇怪现象:0.1加0.1为什么不等于0.2?明...

关键字: 浮点数 编程

在C语言开发中,原生字符串的使用一直存在诸多不便。传统C语言中,字符串本质是以'\0'结尾的固定字符数组,开发人员必须提前预估字符串的最大长度:如果预估过小,拼接或插入字符时会出现缓冲区溢出,引发内存越界错误;如果预估过...

关键字: C语言 字符串

在高并发、低延迟的现代软件系统中,内存管理的效率直接决定了系统的整体性能。传统的动态内存分配方式(如C++中的new/delete、C语言中的malloc/free)虽然使用便捷,但在频繁分配和释放内存的场景下,会产生严...

关键字: C语言 内存

超过 80000 名 IBM 员工正在使用 IBM Bob;平均生产力提升 45%; 多模型编排功能可根据准确性、性能和成本,自动将每个任务路由至合适的模型; 超越代码生成,实现完整的软件开发全生命周期工作...

关键字: IBM 编程 软件开发 AI

在无人机、机器人等智能设备中,九轴IMU(惯性测量单元)是姿态解算的核心传感器,但其原始数据受噪声和零偏影响严重。卡尔曼滤波作为一种基于概率的最优估计方法,通过融合加速度计、陀螺仪和磁力计数据,可显著提升姿态解算的精度与...

关键字: 卡尔曼滤波 九轴IMU C语言

在嵌入式开发中,C语言编写的代码最终会被编译器转化为机器指令,而理解这一转化过程对优化程序性能至关重要。通过反编译工具观察不同优化等级下的汇编代码,开发者能直观看到编译器的"思考方式",从而写出更高效的C代码。

关键字: C语言 反编译工具 编译器

在嵌入式系统开发中,C语言凭借其高效性和接近硬件的特性成为首选语言。然而,这种"贴近硬件"的特性也暗藏危机——内存对齐问题和指针类型转换错误就像隐藏在代码中的定时炸弹,轻则导致性能下降,重则引发硬件异常。本文通过实际案例...

关键字: C语言 嵌入式开发

在单片机开发领域,C语言凭借其高效、易维护和可移植性强的特性,成为了开发者的首选编程语言。而延时程序作为单片机程序中控制时序、协调各模块运行的关键组成部分,其编写的合理性直接影响到整个系统的稳定性与可靠性。然而,看似简单...

关键字: 单片机 C语言

在电子技术飞速发展的当下,单片机作为嵌入式系统的核心部件,广泛应用于工业控制、智能家居、汽车电子等众多领域。对于开发者而言,掌握单片机开发的基本技巧,不仅能提升开发效率,还能优化产品性能、降低成本。

关键字: 单片机 C语言
关闭