一区二区三区三上|欧美在线视频五区|国产午夜无码在线观看视频|亚洲国产裸体网站|无码成年人影视|亚洲AV亚洲AV|成人开心激情五月|欧美性爱内射视频|超碰人人干人人上|一区二区无码三区亚洲人区久久精品

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

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

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

工程師避免這些誤區(qū)才能讓【宏】削鐵如泥

工程師 ? 來源:嵌入式資訊精選 ? 作者:嵌入式資訊精選 ? 2020-10-23 15:15 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

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

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

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

#define strange(file) fprintf (file, “%s %d”,…strange(stderr) p, 35) → fprintf (stderr, “%s %d”, p, 35)

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

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

#define ceil_div(x, y) (x + y - 1) / y

假定其用法如下:

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

拓展開是

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

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

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

如果我們將宏定義為

#define ceil_div(x,y)((x)+(y)-1)/(y)

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

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

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

#define ceil_div(x,y)((((x)+(y)-1)/(y))

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

#define SKIP_SPACES(p, limit) \{ char *lim = (limit); \ while (p 《 lim) { \ if (*p++ != ‘ ’) { \ p--; break; }}}

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

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

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

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

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

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

#define SKIP_SPACES(p, limit) \do { char *lim = (limit); \ while (p 《 lim) { \ if (*p++ != ‘ ’) { \ p--; break; }}} \while (0)

SKIP_SPACES (p, lim);擴展為

do {…} while (0);

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

重復調(diào)用我們常見的“最小”定義一個宏min,如下所示:

#define min(X, Y) ((X) 《 (Y) ? (X) : (Y))

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

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

它擴展如下:

next = ((x + y) 《 (foo (z)) ? (x + y) : (foo (z)));

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

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

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

#define min(X, Y) \({ typeof (X) x_ = (X); \ typeof (Y) y_ = (Y); \ (x_ 《 y_) ? x_ : y_; })

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

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

//假設foo返回int類型#define min(X, Y) ((X) 《 (Y) ? (X) : (Y))…{ int tem = foo (z); next = min (x + y, tem);}

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

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

#define foo (4 + foo)

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

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

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

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

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

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

#define EPERM EPERM

然后宏EPERM擴展為EPERM。實際上,每當在運行文本中使用預處理器時,預處理器都會將其單獨保留。

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

#define x (4 + y)#define y (2 * x)

然后x和y擴展如下:

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

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

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

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

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

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

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

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

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

調(diào)用其他可進行字符串化或連接的宏的宏如果參數(shù)是字符串化或串聯(lián)的,則不會進行預掃描。

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

#define AFTERX(x) X_ ## x#define XAFTERX(x) AFTERX(x)#define TABLESIZE 1024#define BUFSIZE TABLESIZE

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

參數(shù)中使用的宏,其擴展名包含未屏蔽的逗號。這可能導致使用錯誤數(shù)量的參數(shù)調(diào)用在第二次掃描時擴展的宏。舉個例子

#define foo a,b#define bar(x) lose(x)#define lose(x) (1 + (x))

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

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

#define foo (a,b)or#define bar(x) lose((x))

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

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

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

#define ignore_second_arg(a,b,c) a; cignore_second_arg (foo (), ignored (), syntax error);

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

責任編輯:haq

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

    關注

    3

    文章

    4377

    瀏覽量

    64553
  • 編譯器
    +關注

    關注

    1

    文章

    1659

    瀏覽量

    50058
  • 宏匯編器
    +關注

    關注

    0

    文章

    7

    瀏覽量

    9037
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    工程師在電路設計中的八大誤區(qū)

    我們常常會發(fā)現(xiàn),自己想當然的一些規(guī)則或道理往往會存在一些差錯。電子工程師在電路設計中也會有這樣的例子。下面是一位工程師總結(jié)的八大誤區(qū)點。
    發(fā)表于 03-16 17:04 ?1674次閱讀

    電子工程師之路

    ,不應該一直沉默寡言。假如你能正確的明白自己缺少什么條件,怎么去滿足這些條件,我想你基本就要成功了。本人不才,簡單的說下了。本人喜歡交工程師朋友。有愿意和哥們交朋友的還請留言。
    發(fā)表于 11-27 14:02

    工程師手記:FPGA學習的四大誤區(qū)

    工程師手記:FPGA學習的四大誤區(qū)
    發(fā)表于 08-17 23:47

    新手學習PADS的三個誤區(qū)

    很多新手經(jīng)常糾結(jié)的一件事就是學習PADS是不是可以拿高薪,是不是可以成為一個工程師,其實這些都是很大的誤區(qū),誤區(qū)大致都是在以下幾種情況第一:我很熟練的掌握了PADS工具,其實PADS與
    發(fā)表于 07-30 17:10

    如何才能讓自己成為一名嵌入式開發(fā)工程師?

    卻比比皆是。面對一個如此朝陽的行業(yè),我們?nèi)绾?b class='flag-5'>才能讓自己成為一名嵌入式開發(fā)工程師?成為一名優(yōu)秀的嵌入式開發(fā)工程師有哪些要求?  我們都知道,嵌入式主要由硬件和軟件兩部分組成,硬件(微處理器硬件)是軟件
    發(fā)表于 11-08 07:40

    如何才能成為一個硬件工程師?

    如何才能成為一個硬件工程師?
    發(fā)表于 11-11 07:11

    如何才能讓自己成為一名優(yōu)秀的嵌入式開發(fā)工程師

      在互聯(lián)網(wǎng)行業(yè)中,嵌入式技術后來居上,因為物聯(lián)網(wǎng)和人工智能的發(fā)展,嵌入式技術越來越值錢,學嵌入式的人也越來越多。面對一個如此朝陽的行業(yè),我們?nèi)绾?b class='flag-5'>才能讓自己成為一名優(yōu)秀的嵌入式開發(fā)工程師?嵌入式入門
    發(fā)表于 12-17 06:46

    FPGA工程師需要具備哪些技能?

    ,需要具備一系列的技能,才能勝任日益復雜的設計工作。因此,本文將從設計思路、硬件語言、EDA工具、數(shù)字信號處理、通信協(xié)議、測試驗證等多個方面,探討FPGA工程師需要具備哪些技能。 一
    發(fā)表于 11-09 11:03

    工程師故事:怎樣才能成為優(yōu)秀研發(fā)工程師呢?

    工程師故事:怎樣才能成為優(yōu)秀研發(fā)工程師呢?其實三言兩語很難道盡工程師的成長心得。多年的工作中,我們體會比較深刻的有以下幾個方面:
    發(fā)表于 03-22 16:17 ?4250次閱讀

    工程師在電路設計中的八大誤區(qū)

    工程師在電路設計中的八大誤區(qū),感興趣的小伙伴們可以瞧一瞧。
    發(fā)表于 05-12 10:28 ?0次下載

    什么樣的工程師才能被稱為資深Java工程師

    什么樣的工程師才能被稱為資深,又如何成為資深Java工程師呢?
    的頭像 發(fā)表于 08-05 16:15 ?4000次閱讀

    如何才能避免】的這七個誤區(qū)

    當使用參數(shù)調(diào)用時,會將參數(shù)替換為主體,并與其他輸入文件一起檢查結(jié)果,以進行更多的調(diào)用,可以將部分來自主體和部分自變量的調(diào)用組合在一
    的頭像 發(fā)表于 12-24 14:29 ?571次閱讀

    電子工程師設計電路的十四誤區(qū)資料下載

    電子發(fā)燒友網(wǎng)為你提供電子工程師設計電路的十四誤區(qū)資料下載的電子資料下載,更有其他相關的電路圖、源代碼、課件教程、中文資料、英文資料、參考設計、用戶指南、解決方案等資料,希望可以幫助到廣大的電子工程師們。
    發(fā)表于 04-01 08:50 ?22次下載
    電子<b class='flag-5'>工程師</b>設計電路的十四<b class='flag-5'>誤區(qū)</b>資料下載

    一位工程師總結(jié)的電路設計八大誤區(qū)點資料下載

    電子發(fā)燒友網(wǎng)為你提供一位工程師總結(jié)的電路設計八大誤區(qū)點資料下載的電子資料下載,更有其他相關的電路圖、源代碼、課件教程、中文資料、英文資料、參考設計、用戶指南、解決方案等資料,希望可以幫助到廣大的電子工程師們。
    發(fā)表于 04-21 08:52 ?9次下載
    一位<b class='flag-5'>工程師</b>總結(jié)的電路設計八大<b class='flag-5'>誤區(qū)</b>點資料下載