0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

當(dāng)使用參數(shù)調(diào)用宏時,會將參數(shù)替換為宏主體

Q4MP_gh_c472c21 ? 來源:技術(shù)讓夢想更偉大 ? 作者:李肖遙 ? 2020-11-16 16:41 ? 次閱讀

語法錯誤

當(dāng)使用參數(shù)調(diào)用宏時,會將參數(shù)替換為宏主體,并與其他輸入文件一起檢查結(jié)果,以進(jìn)行更多的宏調(diào)用,可以將部分來自宏主體和部分自變量的宏調(diào)用組合在一起。例如,

#definetwice(x)(2*(x)) #definecall_with_1(x)x(1) call_with_1(twice) //x=1 →twice(1) →(2*(1))

宏定義不必帶有括號,通過在宏主體中編寫不平衡的開放括號,可以創(chuàng)建一個從宏主體內(nèi)部開始但在宏主體外部結(jié)束的宏調(diào)用。例如,

#definestrange(file)fprintf(file,"%s%d", … strange(stderr)p,35) →fprintf(stderr,"%s%d",p,35)

組合宏調(diào)用的功能可能會很有用,但是在宏主體中使用不平衡的開放括號只會造成混淆,應(yīng)該避免。

運算符優(yōu)先級問題

在大多數(shù)宏定義示例中,每次出現(xiàn)的宏參數(shù)名稱都帶有括號,并且另一對括號通常會包圍整個宏定義,這是編寫宏最好的方式。舉個例子

#defineceil_div(x,y)(x+y-1)/y

假定其用法如下:

a=ceil_div(b&c,sizeof(int));

拓展開是

a=(b&c+sizeof(int)-1)/sizeof(int);

這沒有達(dá)到我們的預(yù)期,C的運算符優(yōu)先級規(guī)則使其等效于此,而我們想要的是:

a=(((b&c)+sizeof(int)-1))/sizeof(int);

如果我們將宏定義為

#defineceil_div(x,y)((x)+(y)-1)/(y)

可能導(dǎo)致另一種情況,sizeof ceil_div(1,2)是一個C表達(dá)式,可以計算ceil_div(1,2)類型的大小,它擴(kuò)展為:

sizeof((1)+(2)-1)/(2)

這將采用整數(shù)的大小并將其除以2,而除法包含在內(nèi)部的sizeof之外。所以整個宏定義的括號可防止此類問題。那么,下面是定義ceil_div的正確方法如下

#defineceil_div(x,y)((((x)+(y)-1)/(y))

吞噬分號

通常需要定義一個擴(kuò)展為復(fù)合語句的宏。例如,考慮以下宏,該宏跨空格字符前進(jìn)一個指針(參數(shù)p表示在何處查找):

#defineSKIP_SPACES(p,limit) {char*lim=(limit); while(p

該宏定義必須是單個邏輯行,嚴(yán)格來說,該調(diào)用擴(kuò)展為復(fù)合語句,這是一個完整的語句,不需要用分號結(jié)束。

但是,由于它看起來像函數(shù)調(diào)用,因此,如果可以像使用函數(shù)調(diào)用一樣使用它,則可以最大程度地減少混亂,然后再寫一個分號,就像在SKIP_SPACES(p,lim)中一樣。

這可能會在else語句之前出問題,因為分號實際上是空語句。假設(shè)你寫

if(*p!=0) SKIP_SPACES(p,lim); else…

在if條件和else條件之間存在兩個語句(復(fù)合語句和null語句)使C代碼無效。

怎么解決?我們可以使用do…while語句更改宏SKIP_SPACES的定義以解決此問題。方法如下:

#defineSKIP_SPACES(p,limit) do{char*lim=(limit); while(p

SKIP_SPACES (p, lim);擴(kuò)展為

do{…}while(0);

這是一個陳述,循環(huán)僅執(zhí)行一次,而且大多數(shù)編譯器不會為此生成任何額外的代碼。

重復(fù)調(diào)用

我們常見的“最小”定義一個宏min,如下所示:

#definemin(X,Y)((X)

當(dāng)將此宏與包含副作用的參數(shù)一起使用時,如此處所示,

next=min(x+y,foo(z));

它擴(kuò)展如下:

next=((x+y)

其中x + y替換了X,而foo(z)替換了Y。

函數(shù)foo出現(xiàn)在程序中的語句中僅使用一次,但是表達(dá)式foo(z)已兩次替換到宏擴(kuò)展中。結(jié)果,執(zhí)行該語句時可能會兩次調(diào)用foo,所以min是一個不安全的宏。

解決此問題的最佳方法是以僅計算一次foo(z)值的方式定義min。C語言沒有提供執(zhí)行此操作的標(biāo)準(zhǔn)方法,但是可以使用GNU擴(kuò)展來完成此操作,如下所示:

#definemin(X,Y) ({typeof(X)x_=(X); typeof(Y)y_=(Y); (x_

“({{…})”符號產(chǎn)生一個復(fù)合表達(dá)式,它的值是其最后一條語句的值。

如果不使用GNU C擴(kuò)展,唯一的解決方案是在使用宏min時要小心。例如計算foo(z)的值時,將其保存在變量中,然后在min中使用該變量:

//假設(shè)foo返回int類型 #definemin(X,Y)((X)

自引用宏

自引用宏是其名稱出現(xiàn)在其定義中的宏。我們知道所有宏定義都將被重新掃描以查找更多要替換的宏,如果自引用被認(rèn)為是宏的使用,它將產(chǎn)生無限大的擴(kuò)展。

為防止這種情況,自引用不被視為宏調(diào)用。它原樣傳遞到預(yù)處理器輸出中。舉個例子

#definefoo(4+foo)

按照普通規(guī)則,其宏定義分析如下

對foo的每個引用都將擴(kuò)展為(4 + foo);

然后將對其進(jìn)行重新掃描,并將其擴(kuò)展為(4 +(4 + foo));

以此類推,直到計算機(jī)內(nèi)存耗盡。

自引用規(guī)則將這一過程縮短了一步,即(4 + foo),因此此宏定義可能會導(dǎo)致程序在引用foo的任何地方將foo的值加4。

閱讀程序的人看到foo是變量,就難以記得它也是宏,真的會坑爹的。它的一種常見有用用法是創(chuàng)建一個可擴(kuò)展為其自身的宏。如果你寫

#defineEPERMEPERM

然后宏EPERM擴(kuò)展為EPERM。實際上,每當(dāng)在運行文本中使用預(yù)處理器時,預(yù)處理器都會將其單獨保留。

如果宏x擴(kuò)展為使用宏y,而y的擴(kuò)展引用了宏x,則這是x的間接自引用。在這種情況下,x也不展開,舉個例子

#definex(4+y) #definey(2*x)

然后x和y擴(kuò)展如下:

x→(4+y) →(4+(2*x)) y→(2*x) →(2*(4+y))

當(dāng)每個宏出現(xiàn)在另一個宏的定義中時,它們將被展開,但是當(dāng)它間接出現(xiàn)在其自己的定義中時,則不會被展開。

參數(shù)預(yù)掃描處理

宏參數(shù)在被替換為宏主體之前必須經(jīng)過完全宏擴(kuò)展,替換后,將再次掃描整個宏主體,包括替換的參數(shù),以查找要擴(kuò)展的宏。

如果參數(shù)包含任何宏調(diào)用,則它們將在第一次掃描時擴(kuò)展,那么結(jié)果不包含任何宏調(diào)用,因此第二次掃描不會更改它。

如果按照給定的方式替換了參數(shù),并且沒有進(jìn)行預(yù)掃描,則剩余的單個掃描將找到相同的宏調(diào)用并產(chǎn)生相同的結(jié)果。

預(yù)掃描處理在以下三種特殊情況下有大的作用。

對宏的嵌套調(diào)用

當(dāng)宏的參數(shù)包含對該宏的調(diào)用時,就會發(fā)生對宏的嵌套調(diào)用,舉個例子。

如果f是期望一個參數(shù)的宏,則f(f(1))是對f的嵌套調(diào)用對。通過擴(kuò)展f(1)并將其代入f的定義來進(jìn)行所需的擴(kuò)展。預(yù)掃描會導(dǎo)致發(fā)生預(yù)期的結(jié)果。

如果沒有預(yù)掃描,f(1)本身將被替換為參數(shù),并且f的內(nèi)部使用將在主掃描期間作為間接自引用出現(xiàn),并且不會擴(kuò)展。

調(diào)用其他可進(jìn)行字符串化或連接的宏的宏

如果參數(shù)是字符串化或串聯(lián)的,則不會進(jìn)行預(yù)掃描。

如果要擴(kuò)展宏,然后對其擴(kuò)展進(jìn)行字符串化或串聯(lián),則可以通過使一個宏調(diào)用進(jìn)行該字符串化或串聯(lián)的另一宏來實現(xiàn)。舉個例子

#defineAFTERX(x)X_##x #defineXAFTERX(x)AFTERX(x) #defineTABLESIZE1024 #defineBUFSIZETABLESIZE

然后AFTERX(BUFSIZE)擴(kuò)展為X_BUFSIZE,而XAFTERX(BUFSIZE)擴(kuò)展為X_1024而不是X_TABLESIZE,預(yù)掃描始終會進(jìn)行完整的擴(kuò)展。

參數(shù)中使用的宏,其擴(kuò)展名包含未屏蔽的逗號。

這可能導(dǎo)致使用錯誤數(shù)量的參數(shù)調(diào)用在第二次掃描時擴(kuò)展的宏。舉個例子

#definefooa,b #definebar(x)lose(x) #definelose(x)(1+(x))

我們預(yù)期的結(jié)果是bar(foo)變成(1 +(foo)),然后變成(1 +(a,b))。

然而bar(foo)擴(kuò)展為loss(a,b)會出錯,因為Los需要一個參數(shù)。在這種情況下,該問題可以通過使用相同的括號輕松解決,該括號應(yīng)用于防止算術(shù)運算的錯誤嵌套:

#definefoo(a,b) or #definebar(x)lose((x))

多余的一對括號可防止foo定義中的逗號被解釋為參數(shù)分隔符。

參數(shù)中的換行符

類似函數(shù)的宏的調(diào)用可以擴(kuò)展到許多邏輯行,但是在本實施方式中,整個擴(kuò)展是一行完成的。

因此,由編譯器或調(diào)試器發(fā)出的行號是指調(diào)用在其上開始的行,這可能與包含導(dǎo)致問題的參數(shù)的行不同,例如:

#defineignore_second_arg(a,b,c)a;c ignore_second_arg(foo(), ignored(), syntaxerror);

由Syntax error on tokens觸發(fā)的語法錯誤會導(dǎo)致錯誤消息引用第三行(ignore_second_arg行),即使有問題的代碼來自第五行。

責(zé)任編輯:lq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4694

    瀏覽量

    68078
  • 變量
    +關(guān)注

    關(guān)注

    0

    文章

    608

    瀏覽量

    28286
  • 語法
    +關(guān)注

    關(guān)注

    0

    文章

    41

    瀏覽量

    9749

原文標(biāo)題:避免這7個誤區(qū),才能讓【宏】削鐵如泥

文章出處:【微信號:gh_c472c2199c88,微信公眾號:嵌入式微處理器】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    SV中define定義的用法

    SV中使用預(yù)處理指令`define來定義可以用來創(chuàng)建文本替換。根據(jù)場景不同,`define主要用來定義常量、簡化復(fù)雜的表達(dá)式或代碼段以及提高代碼的可移植性。其基本語法為:
    的頭像 發(fā)表于 10-21 14:22 ?46次閱讀

    TINA仿真運放電路時,運放的參數(shù)中是否可以增加溫度參數(shù)呢?

    我們目前在使用TINA仿真運放電路的參數(shù),從模型中找到了外圍電阻是有溫度參數(shù)可以設(shè)定的,包括線性溫度系數(shù),二次溫度系數(shù),指數(shù)溫度系數(shù)。但是運放本身的參數(shù)中目前沒有找到與溫度相關(guān)的參數(shù)
    發(fā)表于 08-14 07:31

    科技擬收購APCB 100%股權(quán)

    科技近期發(fā)布重要公告,宣布其計劃通過全資子公司新加坡勝及PSL,以不超過2.787億元人民幣的現(xiàn)金,全面收購APCB Electronics(Thailand)Co.,Ltd.(簡稱APCB)的100%股權(quán)。此次收購標(biāo)志著勝
    的頭像 發(fā)表于 08-12 15:06 ?437次閱讀

    氣體成功斬獲無錫華潤上華電子大宗載氣項目,進(jìn)軍存量市場

    此次中標(biāo)被視為氣體領(lǐng)域重大突破,標(biāo)志著成熟晶圓產(chǎn)線電子大宗載氣存量業(yè)務(wù)的替換成為可能,從而開拓金電子大宗載氣業(yè)務(wù)新的增長點。
    的頭像 發(fā)表于 04-19 15:27 ?326次閱讀

    鑫科技即將掛牌上市

    浙江鑫科技股份有限公司(以下簡稱“鑫科技”)成功獲得證監(jiān)會的批文,即將掛牌上市,這標(biāo)志著鑫科技將成為資本市場鍛造鋁合金車輪第一股,迎來產(chǎn)業(yè)經(jīng)營和資本運營雙輪驅(qū)動的全新發(fā)展階段。
    的頭像 發(fā)表于 03-11 15:30 ?694次閱讀

    CubeMx生成的stm32f013vet6設(shè)備,對于SDIOCLK頻率設(shè)置的定義與手冊里面對不上是為什么?

    在手冊里面有SDIO adapter clock (SDIOCLK = HCLK) 當(dāng)HCLK設(shè)置為72MHz的時候進(jìn)行配置SDIO_CK, 通過寄存器SDIO_CLKCR的位7:0(CLKDIV
    發(fā)表于 03-08 08:29

    如何利用Rust過程實現(xiàn)derive-with庫呢?

    通過派生 #[derive(With)] 給結(jié)構(gòu)體字段生成 with_xxx 方法,通過鏈?zhǔn)?b class='flag-5'>調(diào)用 with_xxx 方法來構(gòu)造結(jié)構(gòu)體。
    的頭像 發(fā)表于 01-25 09:51 ?258次閱讀

    linux內(nèi)核系統(tǒng)調(diào)用參數(shù)傳遞

    與普通函數(shù)一樣,系統(tǒng)調(diào)用通常需要一些輸入/輸出參數(shù),這些參數(shù)可能包括實際值(即數(shù)字)、用戶模式進(jìn)程地址空間中的變量地址,甚至包括指向用戶模式函數(shù)指針的數(shù)據(jù)結(jié)構(gòu)的地址(參見第11章“信號相關(guān)的系統(tǒng)
    的頭像 發(fā)表于 12-20 09:32 ?1197次閱讀

    C語言-#和##的具體用法

    C語言中,在里面使用’#’和’##’有它非常神奇的作用。在定義的替換的過程中,#號可以作為一個預(yù)處理運算符,把參數(shù)轉(zhuǎn)
    的頭像 發(fā)表于 12-19 12:54 ?3670次閱讀
    C語言-#和##的具體用法

    如何規(guī)范和常量以及命名

    盡量不要使用return、goto、continue、break等改變程序流程的語句。 ◎ 常量建議使用const定義代替,如下 #define ASPECT_RATIO 1.653 替換
    的頭像 發(fā)表于 12-07 14:49 ?633次閱讀

    define定義

    define定義 以#號開頭的都是編譯預(yù)處理指令,它們不是C語言的成分,但是C程序離不開它們,#define用來定義一個,程序在預(yù)處理階段將用define定義的來內(nèi)容進(jìn)行了替換。因此在程序運行時
    的頭像 發(fā)表于 11-24 15:35 ?713次閱讀

    c語言max函數(shù)在哪個庫

    : # define max(a, b) ((a) > (b) ? (a) : (b)) 當(dāng)調(diào)用max函數(shù)時,替換會將函數(shù)
    的頭像 發(fā)表于 11-22 10:18 ?2987次閱讀

    的缺陷與內(nèi)聯(lián)函數(shù)的引入

    雖然有著一定的優(yōu)勢,但是它的缺點也不可忽視。 在編譯階段,我們很難發(fā)現(xiàn)代碼哪里出問題了,因為替換是發(fā)生在預(yù)處理階段,所以有時候在函數(shù)傳參的時候發(fā)生一些錯誤,編譯器不會發(fā)現(xiàn),那它調(diào)
    的頭像 發(fā)表于 11-01 17:57 ?394次閱讀

    預(yù)處理的優(yōu)缺點有哪些

    ? a : b);} 如果我們在程序中將要使用比較大小的函數(shù),我們顯然會選用上面的定義,理由如下: 首先,函數(shù)調(diào)用會帶來額外的開銷,他需要開
    的頭像 發(fā)表于 11-01 17:44 ?353次閱讀

    和函數(shù)應(yīng)該怎么選

    今天我們來看一下利用定義編寫類似函數(shù)調(diào)用的方法和真實的函數(shù)有什么區(qū)別。 一、和函數(shù)怎么選? 首先來看一個例子: # define N 2+2 void main () { int a = N
    的頭像 發(fā)表于 11-01 17:35 ?330次閱讀