良许Linux教程网 干货合集 分享几个嵌入式 C 中的实用技巧

分享几个嵌入式 C 中的实用技巧

1、动态绑定、回调函数

回调函数可实现动态绑定,在一定程度上减少了层与层之间的耦合。关于回调函数,之前我已经撰写过一篇文章:C语言、嵌入式重点知识:回调函数。或许有些初学者还不太理解回调函数,可以参考下图来加深理解:

image-20230908194244685
image-20230908194244685

一般函数调用的顺序都是上层函数(调用者)调用下层函数(被调用者)。而通过上图我们可以看到下层模块的函数2调用了上层模块的函数3,这个调用过程与一般的调用过程相反,这个过程叫做回调,这里上层模块的函数3就是回调函数。回调函数的表现形式是函数指针。

C库stdlib.h中带有一个排序函数:qsort函数。这个排序函数的原型为:

void qsort(void *base, size_t nitems, size_t size, int (*compar)(const void *, const void*));

参数:

  • base— 指向要排序的数组的第一个元素的指针。
  • nitems— 由 base 指向的数组中元素的个数。
  • size— 数组中每个元素的大小,以字节为单位。
  • compar— 用来比较两个元素的函数,即函数指针(回调函数)。
int compar(const void *p1, const void *p2);

如果compar返回值小于0(

如果compar返回值等于0(= 0),那么p1所指向元素与p2所指向元素的顺序不确定;

如果compar返回值大于0(> 0),那么p1所指向元素会被排在p2所指向元素的右面。

例子:

image-20230908194249480
image-20230908194249480
#include 
#include 

int compar_int(const void *p1, const void *p2)
{
 return (*((int*)p1) - *((int*)p2));
}

void test_qsort(void)
{
 int arr[5] = {8, 5, 10, 1, 100};
 
 printf("排序前:");
 for (int i = 0; i printf("%d ", arr[i]);
 }
 
 qsort((int*)arr, 5, 4, compar_int);
 
 printf("\n排序后:");
 for (int i = 0; i printf("%d ", arr[i]);
 }
}

int main(void)
{
 test_qsort();
 return 0;
}

编译、运行结果:

image-20230908194253436
image-20230908194253436

以上就是本次的分享,如有错误,欢迎指出,谢谢。这是第一弹,后续还会继续分享更多实际开发中实用的编程小技巧及编程经验。欢迎持续关注。本文只是盘点了一些实用小技巧,并不是说无论什么场景下都要这么用,还需具体问题具体分析。

2、使用宏给结构体初始化

如果频繁使用一个结构体的话,使用使用宏来给结构体进行赋值是很方便的一种做法。

例子:

#include 

#define  NEW_RECT(length, width)  {(length), (width)}

typedef struct _Rect
{
 int length;
 int width;
}Rect;

int main(void)
{
 Rect rect = NEW_RECT(10, 5);
 printf("rect length = %d, width = %d\n", rect.length, rect.width);
 return 0;
}

编译、运行结果:

image-20230908194256975
image-20230908194256975

这种方法在RT-Thread的底层gpio驱动中也有见到:

image-20230908194300682
image-20230908194300682

3、结构体内置函数指针

我们常常构造一些结构体来存储数据,然后在一些函数中使用这些结构体。下次不妨把数据与操作数据的函数绑在一起,更清晰明了。

相关文章:什么是不完全类型?

例子:

#include 

#define  NEW_RECT(length, width)  {(calc_area), (length), (width)}

typedef struct _Rect
{
 int (*calc_area)(struct _Rect *pThis);
 int length;
 int width;
}Rect;

int calc_area(struct _Rect *pThis)
{
 return (pThis->length * pThis->width);
}

int main(void)
{
 Rect rect = NEW_RECT(10, 5);
 printf("rect length = %d, width = %d\n", rect.length, rect.width);
 printf("rect area = %d\n", rect.calc_area(&rect));
 return 0;
}

编译、运行结果:

image-20230908194305041
image-20230908194305041

4、使用do{}while(0)封装宏

#define DBG_PRINTF(fmt, args...)  \
do\
{\
    printf("> ", __FILE__, __LINE__, __FUNCTION__);\
    printf(fmt, ##args);\
}while(0)

具体说明可查阅往期文章:

C语言、嵌入式中几个非常实用的宏技巧

5、void*

之前在C语言对象编程第一弹:封装与抽象中有介绍过void*void*其实我们平时都有接触过,比如:

void *malloc(size_t size) ;
void *memcpy(void *destin, void *source, unsigned n);
......

void *常常用于函数地封装比较多,当然也有用在其它地方,比如在结构体内定义void*类型的私有指针方便扩展结构体。我们平时在封装自己的函数时,也可以多考虑看看有没有必要使用void*使得函数地通用性更强一些。

以上就是良许教程网为各位朋友分享的Linu系统相关内容。想要了解更多Linux相关知识记得关注公众号“良许Linux”,或扫描下方二维码进行关注,更多干货等着你 !

137e00002230ad9f26e78-265x300
本文由 良许Linux教程网 发布,可自由转载、引用,但需署名作者且注明文章出处。如转载至微信公众号,请在文末添加作者公众号二维码。
良许

作者: 良许

良许,世界500强企业Linux开发工程师,公众号【良许Linux】的作者,全网拥有超30W粉丝。个人标签:创业者,CSDN学院讲师,副业达人,流量玩家,摄影爱好者。
上一篇
下一篇

发表评论

联系我们

联系我们

公众号:良许Linux

在线咨询: QQ交谈

邮箱: yychuyu@163.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

关注微博
返回顶部