當(dāng)前位置:首頁(yè) > 公眾號(hào)精選 > C語(yǔ)言與CPP編程
[導(dǎo)讀]1 概述 在工程規(guī)模較小,不是很復(fù)雜,與硬件結(jié)合緊密,要求移植性的時(shí)候,可采用宏定義簡(jiǎn)化編程,增強(qiáng)程序可讀性。 當(dāng)宏作為常量使用時(shí),C程序員習(xí)慣在名字中只使用大寫字母。但是并沒(méi)有如何將用于其他目的的宏大寫的統(tǒng)一做法。由于宏(特別是帶參數(shù)的宏)可


1 概述

在工程規(guī)模較小,不是很復(fù)雜,與硬件結(jié)合緊密,要求移植性的時(shí)候,可采用宏定義簡(jiǎn)化編程,增強(qiáng)程序可讀性。

當(dāng)宏作為常量使用時(shí),C程序員習(xí)慣在名字中只使用大寫字母。但是并沒(méi)有如何將用于其他目的的宏大寫的統(tǒng)一做法。由于宏(特別是帶參數(shù)的宏)可能是程序中錯(cuò)誤的來(lái)源,所以一些程序員更喜歡使用大寫字母來(lái)引起注意。

  1. 簡(jiǎn)單宏定義

無(wú)參宏的宏名后不帶參數(shù),其定義的一般形式為:

#define 標(biāo)識(shí)符 字符串

// 不帶參數(shù)的宏定義
#define MAX 10

注意:不要在宏定義中放置任何額外的符號(hào),比如"="或者尾部加";"

使用#define來(lái)為常量命名一些優(yōu)點(diǎn):

  • 程序會(huì)更易讀。一個(gè)認(rèn)真選擇的名字可以幫助讀者理解常量的意義;
  • 程序會(huì)更易于修改。我們僅需要改變一個(gè)宏定義,就可以改變整個(gè)程序中出現(xiàn)的所有該常量的值;
  • 可以幫助避免前后不一致或鍵盤輸入錯(cuò)誤;
  • 控制條件編譯;
  • 可以對(duì)C語(yǔ)法做小的修改;
  1. 帶參數(shù)的宏

帶參數(shù)的仍要遵循上述規(guī)則,區(qū)別只是宏名后面緊跟的圓括號(hào)中放置了參數(shù),就像真正的函數(shù)那樣。

#define <宏名>(<參數(shù)列表>) <宏體>

注意參數(shù)列表中的參數(shù)必須是有效的c標(biāo)識(shí)符,同時(shí)以,分隔

算符優(yōu)先級(jí)問(wèn)題:

#define COUNT(M) M*M
int x=5;
print(COUNT(x+1));
print(COUNT(++X));
//結(jié)果輸出:11 和42 而不是函數(shù)的輸出36

注意:

  • 預(yù)編譯器只是進(jìn)行簡(jiǎn)單的文本替換,COUNT(x+1)被替換成COUNT(x+1 x+1),5+15+1=11,而不是36
  • CUNT(++x)被替換成++x*++x即為6 *7=42,而不是想要的6*6=36,連續(xù)前置自加加兩次

解決辦法:

  • 用括號(hào)將整個(gè)替換文本及每個(gè)參數(shù)用括號(hào)括起來(lái)print(COUNT((x+1));

  • 即便是加上括號(hào)也不能解決第二種情況,所以解決辦法是盡量不使用++,-等符號(hào);

分號(hào)吞噬問(wèn)題:

#define foo(x) bar(x); baz(x)

假設(shè)這樣調(diào)用:

if (!feral)
foo(wolf);

將被宏擴(kuò)展為:

if (!feral)
bar(wolf);
baz(wolf);

==baz(wolf);==,不在判斷條件中,顯而易見(jiàn),這是錯(cuò)誤。如果用大括號(hào)將其包起來(lái)依然會(huì)有問(wèn)題,例如

#define foo(x)  { bar(x); baz(x); }
if (!feral)
foo(wolf);
else
bin(wolf);

判斷語(yǔ)言被擴(kuò)展成:

if (!feral) {
bar(wolf);
baz(wolf);
}>>++;++<<
else
bin(wolf);

==else==將不會(huì)被執(zhí)行

解決方法:通過(guò)==do{…}while(0)

#define foo(x)  do{ bar(x); baz(x); }while(0)
if (!feral)
foo(wolf);
else
bin(wolf);

被擴(kuò)展成:

#define foo(x)  do{ bar(x); baz(x); }while(0)
if (!feral)
do{ bar(x); baz(x); }while(0);
else
bin(wolf);

注意:使用do{…}while(0)構(gòu)造后的宏定義不會(huì)受到大括號(hào)、分號(hào)等的影響,總是會(huì)按你期望的方式調(diào)用運(yùn)行。

  1. #運(yùn)算符

#的作用就是將#后邊的宏參數(shù)進(jìn)行字符串的操作,也就是將#后邊的參數(shù)兩邊加上一對(duì)雙引號(hào)使其成為字符串。例如a是一個(gè)宏的形參,則替換文本中的#a被系統(tǒng)轉(zhuǎn)化為"a",這個(gè)轉(zhuǎn)換過(guò)程即為字符串化。

#define TEST(param) #param

char *pStr=TEST(123);
printf("pSrt=%s\n",pStr);
//輸出結(jié)果為字符 ”123“
  1. ##運(yùn)算符

##運(yùn)算符也可以用在替換文本中,它的作用起到粘合的作用,即將兩個(gè)宏參數(shù)連接為一個(gè)數(shù)

#define TEST(param1,param2) (param1##param2)

int num =TEST(13,59);
printf("num=%d\n",num);
//輸出結(jié)果為:num=1359
  1. VA_ARGS

作用主要是為了方便管理軟件中的打印信息。在寫代碼或DEBUG時(shí)通常需要將一些重要參數(shù)打印出來(lái),但在軟件發(fā)行的時(shí)候不希望有這些打印,這時(shí)就用到可變參數(shù)宏了。

 # define PR(...) printf(_VA_ARGS_)
2 PR("hello world\n");
3
4 輸出結(jié)果:hello world

2 一些建議

  • 雖然宏定義很靈活,并且通過(guò)彼此結(jié)合可以產(chǎn)生許多變形用法,但是C++/C程序員不要定義很復(fù)雜的宏,宏定義應(yīng)該簡(jiǎn)單而清晰。
  • 宏名采用大寫字符組成的單詞或其縮寫序列,并在各單詞之間使用“_”分隔。
  • 如果需要公布某個(gè)宏,那么該宏定義應(yīng)當(dāng)放置在頭文件中,否則放置在實(shí)現(xiàn)文件(.cpp)的頂部。
  • 不要使用宏來(lái)定義新類型名,應(yīng)該使用typedef,否則容易造成錯(cuò)誤。
  • 給宏添加注釋時(shí)請(qǐng)使用塊注釋(/* */),而不要使用行注釋。因?yàn)橛行┚幾g器可能會(huì)把宏后面的行注釋理解為宏體的一部分。
  • 盡量使用const取代宏來(lái)定義符號(hào)常量。
  • 對(duì)于較長(zhǎng)的使用頻率較高的重復(fù)代碼片段,建議使用函數(shù)或模板而不要使用帶參數(shù)的宏定義;而對(duì)于較短的重復(fù)代碼片段,可以使用帶參數(shù)的宏定義,這不僅是出于類型安全的考慮,而且也是優(yōu)化與折衷的體現(xiàn)。
  • 盡量避免在局部范圍內(nèi)(如函數(shù)內(nèi)、類型定義內(nèi)等)定義宏,除非它只在該局部范圍內(nèi)使用,否則會(huì)損害程序的清晰性。

3 宏的常見(jiàn)用法

  • 防止一個(gè)頭文件被重復(fù)包含
#ifndef COMDEF_H
#define COMDEF_H
//頭文件內(nèi)容
#endif
  • 得到指定地址上的一個(gè)字節(jié)或字
#define  MEM_B(x) (*((byte *)(x)))
#define MEM_W(x) (*((word *)(x)))
  • 求最大值和最小值
#define  MAX(x,y) (((x)>(y)) ? (x) : (y))
#define MIN(x,y) (((x) < (y)) ? (x) : (y))
  • 得到一個(gè)field在結(jié)構(gòu)體(struct)中的偏移量
#define FPOS(type,field) ((dword)&((type *)0)->field)
  • 得到一個(gè)結(jié)構(gòu)體中field所占用的字節(jié)數(shù)
#define FSIZ(type,field) sizeof(((type *)0)->field)
  • 按照LSB格式把兩個(gè)字節(jié)轉(zhuǎn)化為一個(gè)Word
#define FLIPW(ray) ((((word)(ray)[0]) * 256) + (ray)[1])
  • 得到一個(gè)字的高位和低位字節(jié)
#define WORD_LO(xxx)  ((byte) ((word)(xxx) & 255))
#define WORD_HI(xxx) ((byte) ((word)(xxx) >> 8))
  • 將一個(gè)字母轉(zhuǎn)換為大寫
#define UPCASE(c) (((c)>='a' && (c) <= 'z') ? ((c) – 0×20) : (c))
  • 判斷字符是不是10進(jìn)制的數(shù)字
#define  DECCHK(c) ((c)>='0' && (c)<='9')
  • 判斷字符是不是16進(jìn)制的數(shù)字
#define HEXCHK(c) (((c) >= '0' && (c)<='9') ((c)>='A' && (c)<= 'F') \
((c)>='a' && (c)<='f'))
  • 防止溢出的一個(gè)方法
#define INC_SAT(val) (val=((val)+1>(val)) ? (val)+1 : (val))
  • 返回?cái)?shù)組元素的個(gè)數(shù)
#define ARR_SIZE(a)  (sizeof((a))/sizeof((a[0])))

參考資料

  1. http://www.360doc.com/content/13/0125/13/10906019_262310086.shtml
  2. 高質(zhì)量程序設(shè)計(jì)指南C++/C語(yǔ)言第3版
  3. https://www.cnblogs.com/southcyy/p/10155049.html

點(diǎn)【在看】是最大的支持 

免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺(tái)僅提供信息存儲(chǔ)服務(wù)。文章僅代表作者個(gè)人觀點(diǎn),不代表本平臺(tái)立場(chǎng),如有問(wèn)題,請(qǐng)聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國(guó)汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開(kāi)發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來(lái)越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來(lái)越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開(kāi)幕式在貴陽(yáng)舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語(yǔ)權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營(yíng)業(yè)績(jī)穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤(rùn)率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長(zhǎng) 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營(yíng)商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國(guó)電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長(zhǎng)三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱"軟通動(dòng)力")與長(zhǎng)三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉