这是道哥的第 009 篇原创

一、前言

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

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

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

那么没有想过,当初为什么要扩展出 C++?C 语言有什么样的缺点导致 C++ 的产生?

一步步分析:C语言如何面向对象编程 - 图2

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

但是,C 语言也有很多的优点:

一步步分析:C语言如何面向对象编程 - 图3

其实最后一个优点是最重要的:使用的人越多,生命力就越强。就像现在的社会一样,不是优者生存,而是适者生存。

一步步分析:C语言如何面向对象编程 - 图4

这篇文章,我们就来聊聊如何在 C 语言中利用面向对象的思想来编程。也许你在项目中用不到,但是也强烈建议你看一下,因为我之前在跳槽的时候就两次被问到这个问题。

二、什么是面向对象编程

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

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

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

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

  1. 数据的封装
  2. 继承
  3. 多态

第一个问题:封装

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

第二个问题:继承

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

另外补充一点:学习任何一种语言,一定要理解内存模型!

第三个问题:多态

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

如果一门语言只支持类,而不支持多态,只能说它是基于对象的,而不是面向对象的。

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

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

Animal.h

  1. #ifndef _ANIMAL_H_
  2. #define _ANIMAL_H_
  3. // 定义父类结构
  4. typedef struct {
  5. int age;
  6. int weight;
  7. } Animal;
  8. // 构造函数声明
  9. void Animal_Ctor(Animal *this, int age, int weight);
  10. // 获取父类属性声明
  11. int Animal_GetAge(Animal *this);
  12. int Animal_GetWeight(Animal *this);
  13. #endif

Animal.c

  1. #include "Animal.h"
  2. // 父类构造函数实现
  3. void Animal_Ctor(Animal *this, int age, int weight)
  4. {
  5. this->age = age;
  6. this->weight = weight;
  7. }
  8. int Animal_GetAge(Animal *this)
  9. {
  10. return this->age;
  11. }
  12. int Animal_GetWeight(Animal *this)
  13. {
  14. return this->weight;
  15. }

测试一下:

  1. #include <stdio.h>
  2. #include "Animal.h"
  3. #include "Dog.h"
  4. int main()
  5. {
  6. // 在栈上创建一个对象
  7. Animal a;
  8. // 构造对象
  9. Animal_Ctor(&a, 1, 3);
  10. printf("age = %d, weight = %d \n",
  11. Animal_GetAge(&a),
  12. Animal_GetWeight(&a));
  13. return 0;
  14. }

可以简单的理解为:在代码段有一块空间,存储着可以处理 Animal 对象的函数;在栈中有一块空间,存储着 a 对象。

一步步分析:C语言如何面向对象编程 - 图5

与 C++ 对比: 在 C++ 的方法中,隐含着第一个参数 this 指针。当调用一个对象的方法时,编译器会自动把对象的地址传递给这个指针。

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

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

Dog.h

  1. #ifndef _DOG_H_
  2. #define _DOG_H_
  3. #include "Animal.h"
  4. // 定义子类结构
  5. typedef struct {
  6. Animal parent; // 第一个位置放置父类结构
  7. int legs; // 添加子类自己的属性
  8. }Dog;
  9. // 子类构造函数声明
  10. void Dog_Ctor(Dog *this, int age, int weight, int legs);
  11. // 子类属性声明
  12. int Dog_GetAge(Dog *this);
  13. int Dog_GetWeight(Dog *this);
  14. int Dog_GetLegs(Dog *this);
  15. #endif

Dog.c

  1. #include "Dog.h"
  2. // 子类构造函数实现
  3. void Dog_Ctor(Dog *this, int age, int weight, int legs)
  4. {
  5. // 首先调用父类构造函数,来初始化从父类继承的数据
  6. Animal_Ctor(&this->parent, age, weight);
  7. // 然后初始化子类自己的数据
  8. this->legs = legs;
  9. }
  10. int Dog_GetAge(Dog *this)
  11. {
  12. // age属性是继承而来,转发给父类中的获取属性函数
  13. return Animal_GetAge(&this->parent);
  14. }
  15. int Dog_GetWeight(Dog *this)
  16. {
  17. return Animal_GetWeight(&this->parent);
  18. }
  19. int Dog_GetLegs(Dog *this)
  20. {
  21. // 子类自己的属性,直接返回
  22. return this->legs;
  23. }

测试一下:

  1. int main()
  2. {
  3. Dog d;
  4. Dog_Ctor(&d, 1, 3, 4);
  5. printf("age = %d, weight = %d, legs = %d \n",
  6. Dog_GetAge(&d),
  7. Dog_GetWeight(&d),
  8. Dog_GetLegs(&d));
  9. return 0;
  10. }

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

一步步分析:C语言如何面向对象编程 - 图6

Dog 的内存模型中开头部分就自动包括了 Animal 中的成员,也即是说 Dog 继承了 Animal 的属性。

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

在 C++ 中,如果一个父类中定义了虚函数,那么编译器就会在这个内存中开辟一块空间放置虚表,这张表里的每一个 item 都是一个函数指针,然后在父类的内存模型中放一个虚表指针,指向上面这个虚表。

上面这段描述不是十分准确,主要看各家编译器的处理方式,不过大部分 C++ 处理器都是这么干的,我们可以想这么理解。

子类在继承父类之后,在内存中又会开辟一块空间来放置子类自己的虚表,然后让继承而来的虚表指针指向子类自己的虚表。

一步步分析:C语言如何面向对象编程 - 图7

既然 C++ 是这么做的,那我们就用 C 来手动模拟这个行为:创建虚表和虚表指针。

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

  1. #ifndef _ANIMAL_H_
  2. #define _ANIMAL_H_
  3. struct AnimalVTable; // 父类虚表的前置声明
  4. // 父类结构
  5. typedef struct {
  6. struct AnimalVTable *vptr; // 虚表指针
  7. int age;
  8. int weight;
  9. } Animal;
  10. // 父类中的虚表
  11. struct AnimalVTable{
  12. void (*say)(Animal *this); // 虚函数指针
  13. };
  14. // 父类中实现的虚函数
  15. void Animal_Say(Animal *this);
  16. #endif

2. Animal.c

  1. #include <assert.h>
  2. #include "Animal.h"
  3. // 父类中虚函数的具体实现
  4. static void _Animal_Say(Animal *this)
  5. {
  6. // 因为父类Animal是一个抽象的东西,不应该被实例化。
  7. // 父类中的这个虚函数不应该被调用,也就是说子类必须实现这个虚函数。
  8. // 类似于C++中的纯虚函数。
  9. assert(0);
  10. }
  11. // 父类构造函数
  12. void Animal_Ctor(Animal *this, int age, int weight)
  13. {
  14. // 首先定义一个虚表
  15. static struct AnimalVTable animal_vtbl = {_Animal_Say};
  16. // 让虚表指针指向上面这个虚表
  17. this->vptr = &animal_vtbl;
  18. this->age = age;
  19. this->weight = weight;
  20. }
  21. // 测试多态:传入的参数类型是父类指针
  22. void Animal_Say(Animal *this)
  23. {
  24. // 如果this实际指向一个子类Dog对象,那么this->vptr这个虚表指针指向子类自己的虚表,
  25. // 因此,this->vptr->say将会调用子类虚表中的函数。
  26. this->vptr->say(this);
  27. }

一步步分析:C语言如何面向对象编程 - 图8

在栈空间定义了一个虚函数表 animal_vtbl,这个表中的每一项都是一个函数指针,例如:函数指针 say 就指向了代码段中的函数_Animal_Say()。
对象 a 的第一个成员 vptr 是一个指针,指向了这个虚函数表 animal_vtbl。

3. Dog.h 不变

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

  1. #include "Dog.h"
  2. // 子类中虚函数的具体实现
  3. static void _Dog_Say(Dog *this)
  4. {
  5. printf("dag say \n");
  6. }
  7. // 子类构造函数
  8. void Dog_Ctor(Dog *this, int age, int weight, int legs)
  9. {
  10. // 首先调用父类构造函数。
  11. Animal_Ctor(&this->parent, age, weight);
  12. // 定义子类自己的虚函数表
  13. static struct AnimalVTable dog_vtbl = {_Dog_Say};
  14. // 把从父类中继承得到的虚表指针指向子类自己的虚表
  15. this->parent.vptr = &dog_vtbl;
  16. // 初始化子类自己的属性
  17. this->legs = legs;
  18. }

5. 测试一下

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

内存模型如下:

一步步分析:C语言如何面向对象编程 - 图9

对象 d 中,从父类继承而来的虚表指针 vptr,所指向的虚表是 dog_vtbl。

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

基本上,在 C 中面向对象的开发思想就是以上这样。 这个代码很简单,自己手敲一下就可以了。如果想偷懒,请在后台留言,我发给您。

六、C 面向对象思想在项目中的使用

1. Linux 内核

看一下关于 socket 的几个结构体:

  1. struct sock {
  2. ...
  3. }
  4. struct inet_sock {
  5. struct sock sk;
  6. ...
  7. };
  8. struct udp_sock {
  9. struct sock sk;
  10. ...
  11. };

一步步分析:C语言如何面向对象编程 - 图10

sock 可以看作是父类,inet_sock 和 udp_sock 的第一个成员都是是 sock 类型,从内存模型上看相当于是继承了 sock 中的所有属性。

2. glib 库

以最简单的字符串处理函数来举例:

GString g_string_truncate(GString string, gint len)
GString g_string_append(GString string, gchar val)
GString gstring_prepend(GString _string, gchar val)

API 函数的第一个参数都是一个 GString 对象指针,指向需要处理的那个字符串对象。

  1. GString *s1, *s2;
  2. s1 = g_string_new("Hello");
  3. s2 = g_string_new("Hello");
  4. g_string_append(s1," World!");
  5. g_string_append(s2," World!");

3. 其他项目

还有一些项目,虽然从函数的参数上来看,似乎不是面向对象的,但是在数据结构的设计上看来,也是面向对象的思想,比如:

  1. Modbus 协议的开源库 libmodbus
  2. 用于家庭自动化的无线通讯协议 ZWave
  3. 很久之前的高通手机开发平台 BREW
    https://zhuanlan.zhihu.com/p/338267632