C語言中這些預(yù)定義宏的不可不會
[導(dǎo)讀] 嵌入式er編程,一些常見的宏應(yīng)該掌握如何去使用,會帶給你很多有用的功能。關(guān)于標(biāo)準(zhǔn)定義宏的詳細(xì)示例,見今天推送的第二篇轉(zhuǎn)載文章,總結(jié)的很清晰。本文特別示例了一下如何利用日期時間宏自動內(nèi)置編譯時間信息進(jìn)入固件。
標(biāo)準(zhǔn)預(yù)定義宏
使用示例:
fprintf (stderr, "Internal error: "
"negative string length "
"%d at %s, line %d.",
length, __FILE__, __LINE__);
其他詳細(xì)的示例可以閱讀今日第二篇轉(zhuǎn)載文章,總結(jié)的很清晰很詳細(xì),這里不重復(fù)了。
系統(tǒng)預(yù)定義宏
不同的編譯器或不同的硬件體系結(jié)構(gòu)會定義一些系統(tǒng)相關(guān)的預(yù)定義宏,以IAR for ARM為例:
如何查看宏是否被預(yù)定義呢,我們可以查看編譯器手冊,除此之外,也可以用如下的方式進(jìn)行預(yù)處理提示:
-
#error,用于生成一個編譯錯誤消息,并停止編譯 -
#warning用于生成編譯警告,但不會停止編譯
兩者的語法一樣,如下:
#error "message"
#warning "message"
如:
/*或 error 打印提示,但編譯會停止*/
#ifdef __AAPCS__
#error “__AAPCS__ has been defined"
#endif
/*或 warning 打印提示,但編譯會繼續(xù)*/
#ifdef __AAPCS__
#warning “__AAPCS__ has been defined"
#endif
又比如GNU C,還定義了大量的擴(kuò)展內(nèi)置宏,比如:
如何重定義一個宏
#undef 可以取消已定義的宏,比如:
#ifdef TRUE
#undef TRUE
#endif
#define TRUE 1
這樣做的意義是什么呢?前面已定義了,這里又把它取消掉且再重定義,騷操作?這是有意義且有實用價值的。因為宏是按照包含關(guān)系層層展開的,本質(zhì)是編譯之前的文本替換。這樣就可以明確指定宏在某些部位是確定的值,因為多人協(xié)調(diào)開發(fā),或者使用別的開發(fā)包或者開發(fā)環(huán)境,有的宏已經(jīng)被定義過了,如果不做此處理,這樣宏值很可能不是你想要的值。
這里再強(qiáng)調(diào)一下:宏的本質(zhì)是編譯器之前的文本替換,這也是預(yù)編譯的最重要任務(wù)之一
時間宏的工程實例
項目開發(fā)發(fā)布的固件,經(jīng)常會需要將版本信息、編譯時間信息內(nèi)置在固件中,可能涉及的應(yīng)用需求:
-
進(jìn)行界面顯示 -
上位機(jī)軟件通信讀出該信息 -
固件通過bootloader升級時檢測版本新舊等。
這時就可以直接使用 __DATE__, __TIME__實現(xiàn)將編譯時間自動編譯進(jìn)發(fā)布的固件中。示例代碼如下:
/*IAR for ARM 8.4為例*/
#include <stdio.h>
#define FW_VERSION "1.0.0"
const char *firmwareInf[]={
{__DATE__},
{__TIME__},
{FW_VERSION}
};
int main(void)
{
printf("Compile Time:%s %s \nFw version:%s\n",firmwareInf[0],firmwareInf[1],firmwareInf[2]);
while (1)
{
}
}
來看下療效:
延伸思考,還可以將firmwareInf常量字符串放入一個確定存儲位置,這就需要利用鏈接器腳本來指定存儲位置。
宏的使用在C語言中有極高的價值,如果閱讀u-boot以及Linux內(nèi)核代碼,你會發(fā)現(xiàn)大量的復(fù)雜應(yīng)用,所以有必要先將這些基礎(chǔ)的宏應(yīng)用熟練掌握。
如果覺得本文有價值,在看轉(zhuǎn)發(fā)起來,也算對我的肯定支持。
—END—
長按關(guān)注 |
加群交流 |
免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!