這些乾貨老師是不會教你的,嵌入式 C 中的五大實用技巧分享

2022-04-27     大方老師單片機

原標題:這些乾貨老師是不會教你的,嵌入式 C 中的五大實用技巧分享

這些乾貨老師是不會教你的,嵌入 C中的五大實用技巧分享

///插播一條:我自己在今年年初錄製了一套還比較系統的入門單片機教程,想要的同學找我拿就行了免費的,私信我就可以~點我頭像黑色字體加我地球呺也能領取哦。最近比較閒,帶做畢設,帶學生參加省級或以上比///

正文開始:

1、動態綁定、回調函數

回調函數可以達到動態綁定的作用,在一定程度上可以降低層與層之間的耦合。可能很多初學的小夥伴可能還不理解回調函數,可以藉助下圖來理解:

一般函數調用的順序都是上層函數(調用者)調用下層函數(被調用者)。而通過上圖我們可以看到下層模塊的函2調用了上層模塊的函3,這個調用過程與一般的調用過程相反,這個過程叫做回調,這裡上層模塊的函3就是回調函數。回調函數的表現形式是函數指針。

Cstdlib.h中帶有一個排序函數qsort函數。這個排序函數的原型為:

voidqsort(void *base, size_t nitems, size_t size, int (*compar)(constvoid *, constvoid*));

參數:

·base--指向要排序的數組的第一個元素的指針。

·nitems-- base指向的數組中元素的個數。

·size--數組中每個元素的大小,以位元組為單位。

·compar--用來比較兩個元素的函數,即函數指針(回調函數)。

intcompar(constvoid *p1, constvoid *p2);

compar返回值小0< 0),那p1所指向元素會被排p2所指向元素的左面;

compar返回值等0= 0),那p1所指向元素p2所指向元素的順序不確定;

compar返回值大0>0),那p1所指向元素會被排p2所指向元素的右面。

例子:

#include

#include

intcompar_int(constvoid *p1, constvoid *p2)

{

return (*((int*)p1) - *((int*)p2));

}

voidtest_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]);

}

}

intmain(void)

{

test_qsort();

return0;

}

編譯、運行結果:

以上就是本次的分享,如有錯誤,歡迎指出,謝謝。這是第一彈,後續還會繼續分享更多實際開發中實用的編程小技巧及編程經驗。歡迎持續關注。本文只是盤點了一些實用小技巧,並不是說無論什麼場景下都要這麼用,還需具體問題具體分析。

2、使用宏給結構體初始化

如果頻繁使用一個結構體的話,使用使用宏來給結構體進行賦值是很方便的一種做法。

例子:

#include

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

typedefstruct _Rect

{

int length;

int width;

}Rect;

intmain(void)

{

Rect rect = NEW_RECT(10, 5);

printf("rect length = %d, width = %d\n", rect.length, rect.width);

return0;

}

編譯、運行結果:

這種方法RT-Thread的底gpio驅動中也有見到:

3、結構體內置函數指針

我們常常構造一些結構體來存儲數據,然後在一些函數中使用這些結構體。下次不妨把數據與操作數據的函數綁在一起,更清晰明了。

例子:

#include

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

typedefstruct _Rect

{

int (*calc_area)(struct _Rect *pThis);

int length;

int width;

}Rect;

intcalc_area(struct _Rect *pThis)

{

return (pThis->length * pThis->width);

}

intmain(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));

return0;

}

編譯、運行結果:

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

#define DBG_PRINTF(fmt, args...) \

do\

{\

printf("<> ", __FILE__, __LINE__, __FUNCTION__);\

printf(fmt, ##args);\

}while(0)

5void*

之前有介紹過void*void*其實我們平時都有接觸過,比如:

void *malloc(size_t size) ;

void *memcpy(void *destin, void *source, unsigned n);

......

void *常常用於函數地封裝比較多,當然也有用在其它地方,比如在結構體內定義void*類型的私有指針方便擴展結構體。我們平時在封裝自己的函數時,也可以多考慮看看有沒有必要使用void*使得函數地通用性更強一些。

如果本文對你有幫助,不妨轉發分享,謝~

文章來源: https://twgreatdaily.com/zh-cn/a29a51d441d62cc53ff5542299d83311.html