这是道哥的第009篇原创

一、前言

在嵌入式开发中,C/C++语言是使用最普及的,在C++11版本之前,它们的语法是比较相似的,只不过C++提供了面向对象的编程方式。

虽然C++语言是从C语言发展而来的,但是今天的C++已经不是当年的C语言的扩展了,从2011版本开始,更像是一门全新的语言。

那么没有想过,当初为什么要扩展出C++?C语言有什么样的缺点导致C++的产生?C++在这几个问题上的解决的确很好,但是随着语言标准的逐步扩充,C++语言的学习难度也逐渐加大。没有开发过几个项目,都不好意思说自己学会了C++,那些左值、右值、模板、模板参数、可变模板参数等等一堆的概念,真的不是使用2,3年就可以熟练掌握的。

但是,C语言也有很多的优点:其实最后一个优点是最重要的:使用的人越多,生命力就越强。就像现在的社会一样,不是优者生存,而是适者生存。这篇文章,我们就来聊聊如何在C语言中利用面向对象的思想来编程。也许你在项目中用不到,但是也强烈建议你看一下,因为我之前在跳槽的时候就两次被问到这个问题。

二、什么是面向对象编程

有这么一个公式:程序=数据结构+算法。

C语言中一般使用面向过程编程,就是分析出解决问题所需要的步骤,然后用函数把这些步骤一步一步调用,在函数中对数据结构进行处理(执行算法),也就是说数据结构和算法是分开的。

C++语言把数据和算法封装在一起,形成一个整体,无论是对它的属性进行操作、还是对它的行为进行调用,都是通过一个对象来执行,这就是面向对象编程思想。

如果用C语言来模拟这样的编程方式,需要解决3个问题:数据的封装

继承

多态

第一个问题:封装

封装描述的是数据的组织形式,就是把属于一个对象的所有属性(数据)组织在一起,C语言中的结构体类型天生就支持这一点。

第二个问题:继承

继承描述的是对象之间的关系,子类通过继承父类,自动拥有父类中的属性和行为(也就是方法)。这个问题只要理解了C语言的内存模型,也不是问题,只要在子类结构体中的第一个成员变量的位置放置一个父类结构体变量,那么子类对象就继承了父类中的属性。另外补充一点:学习任何一种语言,一定要理解内存模型!

第三个问题:多态

按字面理解,多态就是“多种状态”,描述的是一种动态的行为。在C++中,只有通过基类引用或者指针,去调用虚函数的时候才发生多态,也就是说多态是发生在运行期间的,C++内部通过一个虚表来实现多态。那么在C语言中,我们也可以按照这个思路来实现。如果一门语言只支持类,而不支持多态,只能说它是基于对象的,而不是面向对象的。

既然思路上没有问题,那么我们就来简单的实现一个。

三、先实现一个父类,解决封装的问题

Animal.h

#ifndef _ANIMAL_H_#define _ANIMAL_H_// 定义父类结构typedef struct {int age;int weight;} Animal;// 构造函数声明void Animal_Ctor(Animal *this, int age, int weight);// 获取父类属性声明int Animal_GetAge(Animal *this);int Animal_GetWeight(Animal *this);#endifAnimal.c#include 'Animal.h'// 父类构造函数实现void Animal_Ctor(Animal *this, int age, int weight){this->age = age;this->weight = weight;}int Animal_GetAge(Animal *this){return this->age;}int Animal_GetWeight(Animal *this){return this->weight;}

测试一下:#include #include 'Animal.h'#include 'Dog.h'int main(){// 在栈上创建一个对象Animal a;// 构造对象Animal_Ctor(&a, 1, 3);printf('age = %d, weight = %d \n',Animal_GetAge(&a),Animal_GetWeight(&a));return 0;}

可以简单的理解为:在代码段有一块空间,存储着可以处理Animal对象的函数;在栈中有一块空间,存储着a对象。与C++对比:在C++的方法中,隐含着第一个参数this指针。当调用一个对象的方法时,编译器会自动把对象的地址传递给这个指针。

所以,在Animal.h中函数我们就模拟一下,显示的定义这个this指针,在调用时主动把对象的地址传递给它,这样的话,函数就可以对任意一个Animal对象进行处理了。

四、 实现一个子类,解决继承的问题

Dog.h#ifndef _DOG_H_#define _DOG_H_#include 'Animal.h'// 定义子类结构typedef struct {Animal parent; // 第一个位置放置父类结构int legs; // 添加子类自己的属性}Dog;// 子类构造函数声明void Dog_Ctor(Dog *this, int age, int weight, int legs);// 子类属性声明int Dog_GetAge(Dog *this);int Dog_GetWeight(Dog *this);int Dog_GetLegs(Dog *this);#endifDog.c#include 'Dog.h'// 子类构造函数实现void Dog_Ctor(Dog *this, int age, int weight, int legs){// 首先调用父类构造函数,来初始化从父类继承的数据Animal_Ctor(&this->parent, age, weight);// 然后初始化子类自己的数据this->legs = legs;}int Dog_GetAge(Dog *this){// age属性是继承而来,转发给父类中的获取属性函数return Animal_GetAge(&this->parent);}int Dog_GetWeight(Dog *this){return Animal_GetWeight(&this->parent);}int Dog_GetLegs(Dog *this){// 子类自己的属性,直接返回return this->legs;}

测试一下:

int main(){Dog d;Dog_Ctor(&d, 1, 3, 4);printf('age = %d, weight = %d, legs = %d \n',Dog_GetAge(&d),Dog_GetWeight(&d),Dog_GetLegs(&d));return 0;}

在代码段有一块空间,存储着可以处理Dog对象的函数;在栈中有一块空间,存储着d对象。由于Dog结构体中的第一个参数是Animal对象,所以从内存模型上看,子类就包含了父类中定义的属性。Dog的内存模型中开头部分就自动包括了Animal中的成员,也即是说Dog继承了Animal的属性。

五、利用虚函数,解决多态问题

在C++中,如果一个父类中定义了虚函数,那么编译器就会在这个内存中开辟一块空间放置虚表,这张表里的每一个item都是一个函数指针,然后在父类的内存模型中放一个虚表指针,指向上面这个虚表。上面这段描述不是十分准确,主要看各家编译器的处理方式,不过大部分C++处理器都是这么干的,我们可以想这么理解。

子类在继承父类之后,在内存中又会开辟一块空间来放置子类自己的虚表,然后让继承而来的虚表指针指向子类自己的虚表。既然C++是这么做的,那我们就用C来手动模拟这个行为:创建虚表和虚表指针。

1. Animal.h为父类Animal中,添加虚表和虚表指针

#ifndef _ANIMAL_H_#define _ANIMAL_H_struct AnimalVTable; // 父类虚表的前置声明// 父类结构typedef struct {struct AnimalVTable *vptr; // 虚表指针int age;int weight;} Animal;// 父类中的虚表struct AnimalVTable{void (*say)(Animal *this); // 虚函数指针};// 父类中实现的虚函数void Animal_Say(Animal *this);#endif

2. Animal.c

#include #include 'Animal.h'// 父类中虚函数的具体实现static void _Animal_Say(Animal *this){// 因为父类Animal是一个抽象的东西,不应该被实例化。// 父类中的这个虚函数不应该被调用,也就是说子类必须实现这个虚函数。// 类似于C++中的纯虚函数。assert(0);}// 父类构造函数void Animal_Ctor(Animal *this, int age, int weight){// 首先定义一个虚表static struct AnimalVTable animal_vtbl = {_Animal_Say};// 让虚表指针指向上面这个虚表this->vptr = &animal_vtbl;this->age = age;this->weight = weight;}// 测试多态:传入的参数类型是父类指针void Animal_Say(Animal *this){// 如果this实际指向一个子类Dog对象,那么this->vptr这个虚表指针指向子类自己的虚表,// 因此,this->vptr->say将会调用子类虚表中的函数。this->vptr->say(this);}在栈空间定义了一个虚函数表animal_vtbl,这个表中的每一项都是一个函数指针,例如:函数指针say就指向了代码段中的函数_Animal_Say()。 > 对象a的第一个成员vptr是一个指针,指向了这个虚函数表animal_vtbl。

3.  Dog.h不变

4. Dog.c中定义子类自己的虚表

#include 'Dog.h'// 子类中虚函数的具体实现static void _Dog_Say(Dog *this){printf('dag say \n');}// 子类构造函数void Dog_Ctor(Dog *this, int age, int weight, int legs){// 首先调用父类构造函数。Animal_Ctor(&this->parent, age, weight);// 定义子类自己的虚函数表static struct AnimalVTable dog_vtbl = {_Dog_Say};// 把从父类中继承得到的虚表指针指向子类自己的虚表this->parent.vptr = &dog_vtbl;// 初始化子类自己的属性this->legs = legs;}

5. 测试一下

int main(){// 在栈中创建一个子类Dog对象Dog d;Dog_Ctor(&d, 1, 3, 4);// 把子类对象赋值给父类指针Animal *pa = &d;// 传递父类指针,将会调用子类中实现的虚函数。Animal_Say(pa);}

内存模型如下:对象d中,从父类继承而来的虚表指针vptr,所指向的虚表是dog_vtbl。

在执行Animal_Say(pa)的时候,虽然参数类型是指向父类Animal的指针,但是实际传入的pa是一个指向子类Dog的对象,这个对象中的虚表指针vptr指向的是子类中自己定义的虚表dog_vtbl,这个虚表中的函数指针say指向的是子类中重新定义的虚函数_Dog_Say,因此this->vptr->say(this)最终调用的函数就是_Dog_Say。

ctor c语言,一步步分析-C语言如何面向对象编程相关推荐

  1. 一步步分析-C语言如何面向对象编程

    这是道哥的第009篇原创 一.前言 在嵌入式开发中,C/C++语言是使用最普及的,在C++11版本之前,它们的语法是比较相似的,只不过C++提供了面向对象的编程方式. 虽然C++语言是从C语言发展而来 ...

  2. 易语言html规则分析,易语言算法原理浅析【一】(示例代码)

    注: 如果你看完了下面的文章.就来试试这个KeyGenMe吧,相信你能有所收获. 一.文章开头首先我们要贴上一段易语言代码,并且编译这段代码,从汇编角度分析易语言程序编译后,易语言算法在汇编中的实现过 ...

  3. r语言npsurv_10生存分析+R语言代码surv.pdf

    10生存分析R语言代码surv 生存函数和危险函数 生存分析数据的 Cox 回归模型 生存分析 R 软件陪同 吴喜之 December 20, 2014 . . . . . . . . . . . . ...

  4. c语言作业的分析,C语言作业分析.doc

    实验一 C程序的运行环境及简单程序编写(一) 一.实验目的 1. 了解VC++6.0集成环境的基本操作方法. 2. 了解在该系统上如何编辑.编译.连接和运行一个C程序. 3. 通过运行简单的C程序,初 ...

  5. C语言做线性分析,C语言版的线性回归分析函数

    前几天,清理出一些十年以前 DOS 下的程序及代码,看来目前也没什么用了,想打个包刻在光碟上,却发现有些代码现在可能还能起作用,其中就有计算一元回归和多元回归的代码,一看代码文件时间,居然是 1993 ...

  6. java语言特点_Java语言特点及分析

    Java语言特点及分析 Java是一门面向对象编程语言. (Java语言作为静态面向对象编程语言的代表,极好地实现了面向对象理论,允许程序员以优雅的思维方式进行复杂的编程.) Java具有简单性.面向 ...

  7. R语言亚组分析 (Subgroup Analysis)及森林图绘制实战

    R语言亚组分析 (Subgroup Analysis)及森林图绘制实战 目录 R语言亚组分析 (Subgroup Analysis)及森林图绘制实战 #亚组分析

  8. R语言伪相关性分析(Spurious Correlation)、相关关系不是因果关系:以哺乳动物数据集msleep为例

    R语言伪相关性分析(Spurious Correlation):相关关系不是因果关系.相关关系不是因果关系.相关关系不是因果关系 #correlation doesn't means causatio ...

  9. R语言伪相关性分析(Spurious Correlation)、相关关系不是因果关系:以缅因州离婚率数据集为例

    R语言伪相关性分析(Spurious Correlation).相关关系不是因果关系:以缅因州离婚率数据集为例 #correlation doesn't means causation 目录

最新文章

  1. 以太坊去中心化_开发以太坊去中心化投票应用程序的指南
  2. 不用数学也能讲清贝叶斯理论的马尔可夫链蒙特卡洛方法?这篇文章做到了
  3. IROS 2021 | 激光视觉融合新思路?Lidar强度图+VPR
  4. Android设置默认文件管理器,使用默认资源管理器在Android中查看文件夹内容
  5. P2216 [HAOI2007]理想的正方形(二维RMQ)
  6. 请给出linux中查看系统已经登录用户的命令?
  7. java 4位数,java 找出4位数的所有吸血鬼数字
  8. 表单提交防止恶意修改
  9. 2008服务器网站设置密码,win2008服务器设置密码
  10. number 限制最长数字_阿博的Python之路Number数据类型详解
  11. 【 Educational Codeforces Round 51 (Rated for Div. 2) F】The Shortest Statement
  12. 超实用:小团队如何从零搭建一个自动化运维体系?
  13. linux clock()_对比python与linux中时间管理的三件工具calender clock datetime
  14. 快速对二叉树和前中后序遍历的相互转化
  15. 电工电子技术计算机用学吗,电工电子技术是学什么
  16. [数字图像处理]频域滤波(2)--高通滤波器,带阻滤波器与陷波滤波器
  17. 转!快速搭建视频直播平台
  18. java+selenium键盘操作
  19. 拆解USB电压电流表,并分析测量原理(转数码之家)测电流需串一小电阻到电路。测电压不用按照文中,可以直接让电压正进AD的输入端口测试
  20. cannot import name ‘mean_squared_erro‘ from ‘sklearn.metrics‘

热门文章

  1. 蓝鲸智云平台部署[6.0.5]
  2. [matlab]三维画图
  3. 纵向数据中抑郁检测与预测的深度多任务学习
  4. 基于VS2015MFC在X86debug编译平台调试opengl 代码出现oxc0000007b错误的解决方法
  5. 网易互娱招聘 | 遇见offer之就要圆你的大厂梦
  6. Arduino智能小车——蓝牙小车
  7. 基础的数组/链表实现的队列
  8. 科技视界杂志科技视界杂志社科技视界编辑部2022年第18期目录
  9. shell编程常用命令总结(二)
  10. 中国佛学66句震撼世界的禅语(转贴)