C语言面向对象设计思想

转自:https://mp.weixin.qq.com/s/afkoDEnVJAn908wtFPB8TQ

面向对象的三个特征:封装,继承,多态。但是C语言不是面向对象编程语言,所以需要借助一些技巧来实现这三个特征:
(1)C语言没有成员函数,struct只能封装数据,不能封装方法,可以在struct里使用函数指针;
(2)C语言不支持继承,可以在一个struct里包含另一个struct;
(3)C语言也没有虚函数,实现多态就更麻烦了,我不会。

举个例子:现在有Point结构体定义如下:

struct Point{
  float x;
  float y;
};

如果想从Point派生出Circle,可以这么写:

struct Circle
{
  struct Point o; //圆心
  float r;        //半径
};

struct Circle C;
struct Point *pPoint = (struct Point*)(&C);

此时如果将Circle类型的指针强制换换成Point类型指针,因为内存是顺序连续的,所以没问题,pPoint->x访问的是C.o.x,pPoint->y访问的是C.o.y.也就是说在需要基类指针的地方可以传入派生类的指针。
但是如果将o和r的顺序换一下就错了。


struct Circle
{
  float r;        //半径
  struct Point o; //圆心
};

struct Circle C;
struct Point *pPoint = (struct Point*)(&C);//这么转会出问题

简单来说就是如果想使用C语言的继承,那么基类对象一定要写在派生类的最前面!但是对C语言编程而言,不建议使用继承和多态,使用封装就可以了。对封装的理解可以退化为:不直接访问结构体的成员变量而是通过函数去访问(C语言没有private属性,直接访问成员变量总是可以的,但是不建议这么做);此外C语言结构体没有this指针,所以使用函数指针封装方法也用的比较少,更多的是提供一些全局函数,将结构体指针作为参数传进去操作。
假设有个Moubus数据包的结构体:


#define PACK_SIZE 256

struct ModbusPack
{
  uint8_t data[PACK_SIZE];
  uint8_t len;
};

还有和它对应的操作函数:

void Modbus_init(struct ModbusPack *pThis);
void Modbus_append(struct ModbusPack *pThis,uint8_t v);
void Modbus_appends(struct ModbusPack *pThis,uint8_t *vs,uint8_t len);
uint8_t Modbus_len(struct ModbusPack *pThis);
void Modbus_append_crc(struct ModbusPack *pThis);
uint8_t Modbus_check(struct ModbusPack *pThis);
void Modbus_init_query(struct ModbusPack *pThis,uint8_t addr,uint8_t fc,uint16_t regBase,uint16_t regNum);
void Modbus_print(struct ModbusPack *pThis);

对于Modbus_len这个函数的实现如下:

uint8_t Modbus_len(struct ModbusPack *pThis) {
  //省略对指针是否为空的判断
  return pThis->len;
}

看起来比直接访问成员变量复杂,好处在于当修改了ModbusPack的实现,将len改为m_len,则只需要修改Modbus_xxx函数就行了,不影响其他地方对这个函数的调用,因为调用的地方只依赖于这个函数的名字,而不需要知道相应结构体的具体实现。


struct ModbusPack
{
  uint8_t m_data[PACK_SIZE];
  uint8_t m_len;
};

uint8_t Modbus_len(struct ModbusPack *pThis) {
  //省略对指针是否为空的判断
  return pThis->m_len;
}

简单的使用例子如下:

#include "modbus.h"

int main()
{
  struct ModbusPack pack;
  Modbus_init(&pack);//C语言没有构造函数,通过初始化函数初始化对象
  Modbus_init_query(&pack,0x01,0x03,0x0000,0x0002);
  Modbus_print(&pack);
  return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值