c语言qsort_C语言、嵌入式项目中一些常用知识及技巧:第一弹

大家好,我是ZhengN。本次给大家分享一些C语言、嵌入式项目中的常用知识。

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

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

例子:

#include <stdio.h>
​
#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 = %dn", rect.length, rect.width);
    return 0;
}

编译、运行结果:

1b5e5f057a29bf9d9df465667a55a183.png

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

3bd14b2577d9fd2b5a8d80832823e23d.png

2、结构体内置函数指针

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

例子:

#include <stdio.h>
​
#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 = %dn", rect.length, rect.width);
    printf("rect area = %dn", rect.calc_area(&rect));
    return 0;
}

编译、运行结果:

fde4ca729e5fed34151918673f08e4c4.png

3、void*

之前在C语言对象编程第一弹:封装与抽象中有用过void*,可以再回去看看。void*其实我们平时都有接触过,比如:

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

void *常常用于函数地封装比较多,当然也有用在其它地方,比如在结构体内定义void*类型的私有指针方便扩展结构体。我们平时在封装自己的函数时,也可以多考虑看看有没有必要使用void*使得函数地通用性更强一些。推荐文章:【C进阶】同事用void把我给秀翻了!

4、动态绑定、回调函数

回调函数可以达到动态绑定的作用,在一定程度上可以降低层与层之间的耦合。关于回调函数,之前已经有写过一篇:C语言、嵌入式重点知识:回调函数。但有很多初学的小伙伴可能还不理解回调函数,可以借助下图来理解:

52845848109d77df109dbce4435dc895.png

一般函数调用的顺序都是上层函数(调用者)调用下层函数(被调用者)。而通过上图我们可以看到下层模块的函数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(< 0),那么p1所指向元素会被排在p2所指向元素的左面;

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

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

例子:

9c75b23b69f005a7bb33ea5ceb3ac0f9.png
#include <stdio.h>
#include <stdlib.h>
​
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 < 5; i++)
    {
        printf("%d ", arr[i]);
    }
    
    qsort((int*)arr, 5, 4, compar_int);
    
    printf("n排序后:");
    for (int i = 0; i < 5; i++)
    {
        printf("%d ", arr[i]);
    }
}
​
int main(void)
{
    test_qsort();
    return 0;
}

编译、运行结果:

1694fe49ed44969a7228c63312ce08aa.png

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

如果本文对你有帮助,欢迎转发分享

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值