声明指向unsigned int类型的对象的指针vptr_一步步分析:C语言如何面向对象编程...

2023-11-14

1b786214b7a2f792783336a3ea69cd9a.png

一、前言

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

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

f1458b1a5aa807ab26ec4908d43bbdac.png

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

16635f598def6d74512c8bd0de1bc474.png

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

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

d6dd9de804554176100c10cdb11d13fb.png

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

03af8b157d39efeffcd1db32986f1d89.png

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

二、什么是面向对象编程

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

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

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

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

  1. 数据的封装
  2. 继承
  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);#endif
Animal.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对象。

2cc02842d3b1e6e7bfd837934c95e150.png

与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);#endif

Dog.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对象,所以从内存模型上看,子类就包含了父类中定义的属性。

f1e502187b7a91df9e508e0fbbfbe32c.png

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

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

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

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

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

8c57787d3e2787b33a0789836491326a.png

既然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);}

58231a64a604ad0d60c694101597dfde.png

在栈空间定义了一个虚函数表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);}

内存模型如下:

57484b5bf3fed0671d0ad51e63e5b3b1.png

对象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的几个结构体:

struct sock {    ...}struct inet_sock {    struct sock sk;    ...};struct udp_sock {    struct sock sk;    ...};

b4471ea4a6c94269ecef04a36fc032b4.png

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 *g_string_prepend(GString *string, gchar *val)

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

GString *s1, *s2;s1 = g_string_new("Hello");s2 = g_string_new("Hello");g_string_append(s1," World!");g_string_append(s2," World!");

3. 其他项目

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

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


【原创声明】

作者:道哥(公众号: IOT物联网小镇)
知乎:道哥
B站:道哥分享
掘金:道哥分享
CSDN:道哥分享

END 来源:IoT物联网小镇,作者:道哥
版权归原作者所有,如有侵权,请联系删除。 ▍ 推荐阅读 国内MCU能替代国外产品吗?MCU的未来又将如何? STM32价格疯长下,盘点STM32的国产替代者 选微处理器MPU,还是单片机MCU?两者区别详解

e6977fd82fe42c010e16d58d186843c3.gif

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

声明指向unsigned int类型的对象的指针vptr_一步步分析:C语言如何面向对象编程... 的相关文章

  • 转:在线HTML编辑器 CKEditor4 下载及使用说明。

    转 在线HTML编辑器 CKEditor4 下载及使用说明 这里是CKEditor4最新版本 演示实例 其它实例请下载后可查看源码 CKEditor4是FCKeditor之后重新开发的版本 是一款专门使用在网页上属于开放源代码的所见即所得文
  • 3.6设计模式————迪米特法则——面向对象设计原则

    迪米特法则的定义 迪米特法则 Law of Demeter LoD 又叫作最少知识原则 Least Knowledge Principle LKP 产生于 1987 年美国东北大学 Northeastern University 的一个名为
  • Postgresql:删除及查询字段中包含单引号的数据

    Postgresql 删除及查询字段中包含单引号的数据 1 假设pg表t info的属性att 值为固定的 test 2 假设值为不固定的 abcde 参考 1 假设pg表t info的属性att 值为固定的 test 删除可以用 用 转义
  • Java常用类(二):StringUtils类

    目录 1 什么是StringUtils类 2 StringUtils类的常用方法 2 1 isEmpty String str 2 2 isNotEmpty String str 2 3 isBlank String str 2 4 isN
  • C/C++库函数(tolower/toupper)实现字母的大小写转换

    C C 库函数 tolower toupper 实现字母的大小写转换 本文将介绍库函数实现字母的大小写转换 常用到的是在ctype h C 中是cctype 库文件下定义的函数方法 首先来看一下C下tolower toupper函数实现原型
  • latex:圆周率如何表示

    2019独角兽企业重金招聘Python工程师标准 gt gt gt pi 转载于 https my oschina net letiantian blog 316387
  • Python常见排序之冒泡排序(详解)

    一 为啥叫冒泡排序呢 因为越小的元素会经由交换慢慢 浮 到数列的顶端 升序或降序排列 就如同碳酸饮料中co2的气泡最终会上浮到顶端一样 故名 冒泡排序 二 排序思路 1 相邻元素之间的比较 如果第一个元素比第二个元素大 则他们两个进行交换
  • java中实现es count distinct

    需求 说一下最近项目中碰到的一个es方面的需求 在一个字段columnB里面 获取特定值0的总数据量 然后在这些数据里面 再做另一个字段columnA的去重并得到该字段值的集合 方案 两种方案写成sql类似如下 两次es查询 SELECT
  • 【论文精度】Transformer--Attention Is All You Need

    沐神论文精度 https www bilibili com video BV1pu411o7BE spm id from pageDriver Paper https arxiv org pdf 1706 03762 pdf Transfo
  • 【cocos2d-x 2】关于CocosStudio编辑出的csb的简单使用

    说明 cocos2d x版本为 3 4 cocosStudio版本为2 1 5 cocosStudio 我个人目前的认识 以为就是一个界面编辑器 界面通过美工可以排布好 然后把发布的csb和png资源文件 提供给程序员使用 1 关于 coc
  • 大语言模型之八-提示工程(Prompt engineer)

    大语言模型的效果好 很大程度上归功于算力和数据集 算力使得模型可以足够大 大到模型的理解 记忆 推理和生成以及泛化能力可以同时兼备 而数据集则是模型学习的来源 LLM中的prompt方法主要用于fine tune阶段 即在预训练的基础上 针
  • Pyramid Scene Parsing Network

    Pyramid Scene Parsing Network Keras实现代码链接 https github com BBuf Keras Semantic Segmentation Contribution 提出了金字塔场景解析网络 以将
  • poj 2155 Matrix

    Problem poj org problem id 2155 vjudge net contest 146952 problem A Referencd www cnblogs com gj Acit p 3258880 html Mea
  • mysql Heartbeat主主同步方案

    Heartbeat高可用Mysql主主同步方案 1 1 方案简介 本方案使用heartbeat mysql主主同步来实现mysql数据库的高可用 当服务器或者master的heartbeat宕掉以后会自动切换到backup上 服务器或者ma
  • 使用MATLAB进行多元线性回归预测

    在回归分析中 如果有两个或两个以上的自变量 就称为多元回归 事实上 一种现象常常是与多个因素相联系的 由多个自变量的最优组合共同来预测或估计因变量 比只用一个自变量进行预测或估计更有效 更符合实际 因此多元线性回归比一元线性回归的实用意义更
  • B站马士兵python入门基础版详细笔记(7)

    B站马士兵python入门基础版详细笔记 7 接着这个博客写的笔记 他写的还是很详细的 但是只写到了第六章 python基础知识 记录在B站视频看到的python的基础知识 我的笔记就有点偷懒了 很多没有敲代码 直接截的图 一 字典 列表
  • 【半监督学习】2、Soft Teacher

    文章目录 一 背景 二 方法 2 1 End to End Pseudo Labeling Framework 2 2 Soft teacher 2 3 Box Jittering 三 实验 论文 End to End Semi Super
  • C++ void** 类型

    在C 中 0 或者字面值 nullptr 能转换成任意指针类型 指向任意非常量的指针能转换成 void 指向任意常量对象的指针能转换成 const void 参考 C 类型转换 所以 void 类型可以用以下方式理解 void ppDefi
  • 在docusaurus中使用Vue组件⚡

    docusaurus 类似于Vuepress 是一个静态站点生成工具 但不同的是 使用docusaurus可以在文档中轻易混入React组件 很适合写React相关组件的文档 而Vuepress则是适合混入Vue组件 这就让我比较困惑 如果

随机推荐

  • error: #35: #error directive: "... 解决方法

    出现 USER stm32f10x h 96 error 35 error directive Please select first the target STM32F10x device used in your application
  • shell大小中括号作用

    http taotaocoder blog 163 com blog static 200228274201110306203483 Shell的强大是毋庸置疑的 方便了我们也迷惑了我们 比如这些杂七杂八的括号 一向自认聪明的我也傻傻分不清
  • 动态鼠标指针_推荐8款电脑鼠标指针,让你电脑不再千篇一律

    今天应该朋友给我留言说想要一款电脑指针 我就去搜罗了八款电脑指针 这些电脑指针有优美的 有酷炫的 有简洁的 可盐可甜 01 Breeze Cursors样式 002 a element 样式3D动态旋转鼠标指针 003 bibata cur
  • Self-study Python Fish-C Note-2 P9-P14

    先存一个网址非常详细的介绍python的数据类型 https www runoob com python3 python3 data type html A 数字 Numbers 1 整数 integers python 的整数长度是不受限
  • 类中的成员

    类中的成员 字段 方法 属性 1 字段 字段 普通字段 静态字段 1 class Mycalss 2 str 这是静态字段 3 def init self str1 4 self str1 str1 普通字段 5 实例化对象 6 a Myc
  • php提交表单显示错误,php – 在提交注册表单时使用jQuery显示错误

    你需要修好几件事情 gt 首先 处理注册过程的文件不应该是与表单相同的文件 gt 它纯粹用于处理数据 因此不能使用头 Location login php 直接重定向浏览器 这部分应该由你的JavaScript代码来处理 gt 您还需要告诉
  • 笔记--java sort() 方法排序

    背景 最近在刷一道算法题 字符串重新排序 时 发现自己有思路但是写代码的时候就无从下手了 而且看了答案之后还没看懂 关键就是基础不好 对于排序没有理解 虽然我学过常用的排序算法 但是都是理念 实践少 目的 从实践和原理出发 重点是从实践出发
  • 基于博弈搜索算法的智能五子棋设计

    基于博弈搜索算法的智能五子棋设计 0 引言 在智能过程中 搜索是必不可少的 是人工智能中的一个基本问题 这是因为人工智能研究的主要是那些没有成熟方法可依的问题领域 需要一步一步搜索求解 游戏中如何找到对自己有利的局面就属于这类问题 在游戏
  • SQL注入式攻击

    一 SQL注入式攻击 1 所谓SQL注入式攻击 就是攻击者把SQL命令插入到Web表单的输入域或页面请求的查询字符串 欺骗服务器执行恶意的SQL命令 2 在某些表单中 用户输入的内容直接用来构造 或者影响 动态SQL命令 或作为存储过程的输
  • CS-Stdio Display Builder

    Display Builder 1 操作界面编辑器和Runtime 2 在EPICS edd dm medm edm 想法上构建 3 与CS Studio BOY 兼容性非常好 4 大约2015年在CS Stdio Eclipse中开始 现
  • 【高阶】linux内核环形缓冲区ring buffer实现原理分析

    1 前言 最近项目中用到一个环形缓冲区 ring buffer 代码是由linux内核的kfifo改过来的 缓冲区在文件系统中经常用到 通过缓冲区缓解cpu读写内存和读写磁盘的速度 例如一个进程A产生数据发给另外一个进程B 进程B需要对进程
  • 非root用户安装expect

    非root用户安装expect 文章目录 非root用户安装expect 准备 tcl安装 expect安装 怎么卸载 看了很多教程 踩了很多坑 分享给需要的人 准备 参考资料 ubuntu安装expect 其实不用root权限的原理很简单
  • 设置VS 2013代码折叠

    设置VS 2013代码折叠 版本 VS2013 中文版 步骤 工具 T gt 选项 O gt 文本编辑器 gt C C gt 查看 gt 大纲语句块 设置为 True
  • JavaScript基础

  • matlab低通滤波器

    clc 清除命令窗口 clear 清除所有变量 close all 关闭所有的图形窗口 N 2 10 定义一个变量N 值为2的10次方 n 0 N 1 生成一个从0到N 1的序列 Fs 2000 采样频率 tn n 1 Fs 时间序列 Fn
  • 为什么需要单元测试?

    为什么需要单元测试 从产品角度而言 常规的功能测试 系统测试都是站在产品局部或全局功能进行测试 能够很好地与用户的需要相结合 但是缺乏了对产品研发细节 特别是代码细节的理解 从测试人员角度而言 功能测试和系统测试以及其他性能测试等等对测试人
  • Windows下忘记MySQL root密码解决方法

    Windows下忘记MySQL密码的解决办法网上好多好多 可是 我发现 如果采用Windows服务启动的时候 安装网上通过命令行修改root密码的方法行不通 经过实验 发现 Windows的服务运行的配置并不是在命令行下的配置 查看Wind
  • anaconda怎么运行python脚本_Anaconda运行python脚本 Anaconda方法教程

    你是否想了解Anaconda运行python脚本的操作 下面就是笔者带来的Anaconda运行python脚本的操作步骤 赶紧来看一下吧 相信对大家一定会有所帮助哦 Anaconda是使用 虚拟 环境里边运行Python 这样便于版本 包管
  • 面向对象的设计思想

    面向对象的设计思想 OO思想 Object Oriented 1 看到一个需求的时候不应该直接写代码 应该先考虑有哪些类 2 考虑类的时候 类一定是一类事务的描述 不能太局限 3 考虑类的时候需要考虑主要的类 也就是需要和业务 动作 事件紧
  • 声明指向unsigned int类型的对象的指针vptr_一步步分析:C语言如何面向对象编程...

    一 前言 在嵌入式开发中 C C 语言是使用最普及的 在C 11版本之前 它们的语法是比较相似的 只不过C 提供了面向对象的编程方式 虽然C 语言是从C语言发展而来的 但是今天的C 已经不是当年的C语言的扩展了 从2011版本开始 更像是一