㈠ C語言編譯系統對宏替換的處理是在什麼時候進行的
是正式工作開始之前的准備工作,所以宏替換是在對程序編譯之前進行的。
宏替換是C/C++的預處理中的一部分,對於宏定義中的形參,在替換列表中,如果不是作為#或##的操作數,那麼將對應實參完全展開(相當於對實參進行求值),然後將替換列表中的形參替換掉,如果是#或##的操作數。
(1)宏替換在編譯過程中的哪一階段擴展閱讀:
宏的用途在於自動化頻繁使用的序列或者是獲得一種更強大的抽象能力。
計算機語言如C語言或匯編語言有簡單的宏系統,由編譯器或匯編器的預處理器實現。C語言的宏預處理器的工作只是簡單的文本搜索和替換,使用附加的文本處理語言如M4,C程序員可以獲得更精巧的宏。
宏的行為如同是函數對自身程序文本的變形,並且可以應用全部語言來表達這種變形。一個C宏可以定義一段語法的替換,然而一個Lisp的宏卻可以控制一節代碼的計算。
㈡ C語言中,下面有關宏替換的敘述中,不正確的是
D 宏替換是在預編譯時執行的
㈢ C語言文件的編譯與執行的四個階段並分別描述
開發C程序有四個步驟:編輯、編譯、連接和運行。
任何一個體系結構處理器上都可以使用C語言程序,只要該體系結構處理器有相應的C語言編譯器和庫,那麼C源代碼就可以編譯並連接到目標二進制文件上運行。
1、預處理:導入源程序並保存(C文件)。
2、編譯:將源程序轉換為目標文件(Obj文件)。
3、鏈接:將目標文件生成為可執行文件(EXE文件)。
4、運行:執行,獲取運行結果的EXE文件。
(3)宏替換在編譯過程中的哪一階段擴展閱讀:
將C語言代碼分為程序的幾個階段:
1、首先,源代碼文件測試。以及相關的頭文件,比如stdio。H、由預處理器CPP預處理為.I文件。預編譯的。文件不包含任何宏定義,因為所有宏都已展開,並且包含的文件已插入。我歸檔。
2、編譯過程是對預處理文件進行詞法分析、語法分析、語義分析和優化,生成相應的匯編代碼文件。這個過程往往是整個程序的核心部分,也是最復雜的部分之一。
3、匯編程序不直接輸出可執行文件,而是輸出目標文件。匯編程序可以調用LD來生成可以運行的可執行程序。也就是說,您需要鏈接大量的文件才能獲得「a.out」,即最終的可執行文件。
4、在鏈接過程中,需要重新調整其他目標文件中定義的函數調用指令,而其他目標文件中定義的變數也存在同樣的問題。
㈣ C語言中,宏替換的替換規則
簡單來說:宏定義又稱為宏代換、宏替換,簡稱「宏」。是C提供的三種預處理功能的其中一種。
復雜的請看下面,講的很全。下面的帶參宏定義,多行宏定義,在Linux內核源碼中很多。另外sizeof也是一個宏定義。
宏定義
宏定義是C提供的三種預處理功能的其中一種,這三種預處理包括:宏定義、文件包含、條件編譯
1. 不帶參數的宏定義:
宏定義又稱為宏代換、宏替換,簡稱「宏」。
格式: #define 標識符 字元串
其中的標識符就是所謂的符號常量,也稱為「宏名」。
預處理(預編譯)工作也叫做宏展開:將宏名替換為字元串。
掌握"宏"概念的關鍵是「換」。一切以換為前提、做任何事情之前先要換,准確理解之前就要「換」。
即在對相關命令或語句的含義和功能作具體分析之前就要換:
例: #define PI 3.1415926 把程序中出現的PI全部換成3.1415926
說明:
(1)宏名一般用大寫
(2)使用宏可提高程序的通用性和易讀性,減少不一致性,減少輸入錯誤和便於修改。例如:數組大小常用宏定義
(3)預處理是在編譯之前的處理,而編譯工作的任務之一就是語法檢查,預處理不做語法檢查。
(4)宏定義末尾不加分號;
(5)宏定義寫在函數的花括弧外邊,作用域為其後的程序,通常在文件的最開頭。
(6)可以用#undef命令終止宏定義的作用域
(7)宏定義可以嵌套
(8)字元串" "中永遠不包含宏
(9)宏定義不分配內存,變數定義分配內存。
2. 帶參數的宏定義:
除了一般的字元串替換,還要做參數代換
格式: #define 宏名(參數表) 字元串
例如:#define S(a,b) a*b
area=S(3,2);第一步被換為area=a*b; ,第二步被換為area=3*2;
類似於函數調用,有一個啞實結合的過程:
(1)實參如果是表達式容易出問題
#define S(r) r*r
area=S(a+b);第一步換為area=r*r;,第二步被換為area=a+b*a+b;
正確的宏定義是#define S(r) ((r)*(r))
(2)宏名和參數的括弧間不能有空格
(3)宏替換只作替換,不做計算,不做表達式求解
(4)函數調用在編譯後程序運行時進行,並且分配內存。宏替換在編譯前進行,不分配內存
(5)宏的啞實結合不存在類型,也沒有類型轉換。
(6)函數只有一個返回值,利用宏則可以設法得到多個值
(7)宏展開使源程序變長,函數調用不會
(8)宏展開不佔運行時間,只佔編譯時間,函數調用占運行時間(分配內存、保留現場、值傳遞、返回值)
3. 宏定義其他冷門、重點知識
#define用法
1、 用無參宏定義一個簡單的常量
#define LEN 12
這個是最常見的用法,但也會出錯。
比如下面幾個知識點你會嗎?可以看下:
(1) #define NAME "zhangyuncong"
程序中有"NAME"則,它會不會被替換呢?
(2) #define 0x abcd
可以嗎?也就是說,可不可以用把標識符的字母替換成別的東西?
(3) #define NAME "zhang
這個可以嗎?
(4) #define NAME "zhangyuncong"
程序中有上面的宏定義,並且,程序里有句:
NAMELIST這樣,會不會被替換成"zhangyuncong"LIST
四個題答案都是否定的。
第一個,""內的東西不會被宏替換。這一點應該大都知道。
第二個,宏定義前面的那個必須是合法的用戶標識符
第三個,宏定義也不是說後面東西隨便寫,不能把字元串的兩個""拆開。
第四個:只替換標識符,不替換別的東西。NAMELIST整體是個標識符,而沒有NAME標識符,所以不替換。
也就是說,這種情況下記住:#define 第一位置第二位置
(1) 不替換程序中字元串里的東西。
(2) 第一位置只能是合法的標識符(可以是關鍵字)
(3) 第二位置如果有字元串,必須把""配對。
(4) 只替換與第一位置完全相同的標識符
還有就是老生常談的話:記住這是簡單的替換而已,不要在中間計算結果,一定要替換出表達式之後再算。
2、 帶參宏一般用法
比如#define MAX(a,b) ((a)>(b)?(a):(b))
則遇到MAX(1+2,value)則會把它替換成:
((1+2)>(value)?(1+2):(value))
注意事項和無參宏差不多。
但還是應注意
#define FUN(a) "a"
則,輸入FUN(345)會被替換成什麼?
其實,如果這么寫,無論宏的實參是什麼,都不會影響其被替換成"a"的命運。
也就是說,""內的字元不被當成形參,即使它和一模一樣。
那麼,你會問了,我要是想讓這里輸入FUN(345)它就替換成"345"該怎麼實現呢?
請看下面關於#的用法
3、 有參宏定義中#的用法
#define STR(str) #str
#用於把宏定義中的參數兩端加上字元串的""
比如,這里STR(my#name)會被替換成"my#name"
一般由任意字元都可以做形參,但以下情況會出錯:
STR())這樣,編譯器不會把「)」當成STR()的參數。
STR(,)同上,編譯器不會把「,」當成STR的參數。
STR(A,B)如果實參過多,則編譯器會把多餘的參數捨去。(VC++2008為例)
STR((A,B))會被解讀為實參為:(A,B),而不是被解讀為兩個實參,第一個是(A第二個是B)。 4、 有參宏定義中##的用法
#define WIDE(str) L##str
則會將形參str的前面加上L
比如:WIDE("abc")就會被替換成L"abc"
如果有#define FUN(a,b) vo##a##b()
那麼FUN(id ma,in)會被替換成void main()
5、 多行宏定義:
#define doit(m,n) for(int i=0;i<(n);++i)\
{\
m+=i;\
}
㈤ C語言中宏替換占不佔用運行時間
宏替換,發生在預編譯的時候,預編譯後才能編譯運行,在運行的時候,宏替換已經完成,故不會佔用運行時間
㈥ 在下列有關宏替換的敘述中,正確的說法是 A)宏替換佔用運行時間
宏名用大寫字母表示只是個習慣,大小寫都行;
宏替換不佔用運行時間,只佔編譯時間;因為宏替換發生在編譯預處理階段,預處理也是編譯工作的一部分。
宏名無類型;
宏替換只是字元替換;
帶參數的宏替換和函數不等價;
所以答案選D。
㈦ C宏定義在什麼時候替換
1 作用時間不同。 宏定義在編譯期間即會使用並替換,而全局變數要到運行時才可以。 2 本質類型不同。 宏定義的只是一段字元,在編譯的時候被替換到引用的位置。在運行中是沒有宏定義的概念的。
㈧ 以下有關宏替換和函數的調用的敘述中,不正確的是
選A
宏替換只在編譯階段發生,不在運行階段發生
㈨ 求解釋C語言的宏定義
根據你上面的定義,s(a+b)等價於:3.5*a+b*a+b,這應該不是你想要的吧
所以,定義宏時要加括弧,應該這樣:
#dewfine s(x) pt*(x)*(x)
㈩ C語言 四個過程:預處理,編譯,匯編,鏈接,分別進行了什麼過程別度娘。
1.預處理 這個階段用來處理所有的預處理指令 - 主要是一些宏替換,頭文件導入
2.編譯 對預處理後的代碼進行翻譯工作,得到計算機
所認識的格式編譯工作得到的結果文件叫做目標文件,
擴展名是.o(編譯得到匯編文件(.s) ->經過匯編後得到.o目標文件)
3.鏈接 把所有的目標文件和其他必要的文件合並在一起
得到最終的可執行文件