c語言宏定義

一.?
#define是C語言中提供的宏定義命令,其主要目的是為程序員在編程時提供一定的方便,并能在一定程度上提高程序的運行效率,但學生在學習時往往不能理解該命令的本質,總是在此處產生一些困惑,在編程時誤用該命令,使得程序的運行與預期的目的不一致,或者在讀別人寫的程序時,把運行結果理解錯誤,這對 C語言的學習很不利。?
1 #define命令剖析?
1.1?? #define的概念?
#define命令是C語言中的一個宏定義命令,它用來將一個標識符定義為一個字符串,該標識符被稱為宏名,被定義的字符串稱為替換文本。?
該命令有兩種格式:一種是簡單的宏定義,另一種是帶參數的宏定義。?
(1)?? 簡單的宏定義:?
#define?? <宏名>  <字符串>?
?? 例:?? #define PI 3.1415926?
(2) 帶參數的宏定義?
?? #define?? <宏名> (<參數表>)?? <宏體>?
?? 例: #define?? A(x) x?
一個標識符被宏定義后,該標識符便是一個宏名。這時,在程序中出現的是宏名,在該程序被編譯前,先將宏名用被定義的字符串替換,這稱為宏替換,替換后才進行編譯,宏替換是簡單的替換。?
1.2 宏替換發生的時機?
為了能夠真正理解#define的作用,讓我們來了解一下對C語言源程序的處理過程。當我們在一個集成的開發環境如Turbo C中將編寫好的源程序進行編譯時,實際經過了預處理、編譯、匯編和連接幾個過程,見圖1。?

源程序?

預處理器?

修改后的源程序?

編譯器?

匯編程序?

匯編器?

可重定位的目標程序?

連接器?

可執行的目標程序?

圖1 C語言的編譯過程?
其中預處理器產生編譯器的輸出,它實現以下的功能:?
(1)??? 文件包含?
可以把源程序中的#include 擴展為文件正文,即把包含的.h文件找到并展開到#include 所在處。?
(2)??? 條件編譯?
預處理器根據#if和#ifdef等編譯命令及其后的條件,將源程序中的某部分包含進來或排除在外,通常把排除在外的語句轉換成空行。?
(3)??? 宏展開?
預處理器將源程序文件中出現的對宏的引用展開成相應的宏 定義,即本文所說的#define的功能,由預處理器來完成。?
經過預處理器處理的源程序與之前的源程序有所有不同,在這個階段所進行的工作只是純粹的替換與展開,沒有任何計算功能,所以在學習#define命令時只要能真正理解這一點,這樣才不會對此命令引起誤解并誤用。?

2 #define使用中的常見問題解析?
2.1 簡單宏定義使用中出現的問題?
在簡單宏定義的使用中,當替換文本所表示的字符串為一個表達式時,容易引起誤解和誤用。如下例:?
例1?? #define?? N?? 2+2?
void main()?
{?
?? int?? a=N*N;?
?? printf(“%d”,a);?
}?
(1) 出現問題:在此程序中存在著宏定義命令,宏N代表的字符串是2+2,在程序中有對宏N的使用,一般同學在讀該程序時,容易產生的問題是先求解N為2+2=4,然后在程序中計算a時使用乘法,即N*N=4*4=16,其實該題的結果為8,為什么結果有這么大的偏差??
(2)問題解析:如1節所述,宏展開是在預處理階段完成的,這個階段把替換文本只是看作一個字符串,并不會有任何的計算發生,在展開時是在宏N出現的地方只是簡單地使用串2+2來代替N,并不會增添任何的符號,所以對該程序展開后的結果是a=2+2*2+2,計算后=8,這就是宏替換的實質,如何寫程序才能完成結果為16的運算呢??
(3)解決辦法:將宏定義寫成如下形式?
#define?? N?? (2+2)?
這樣就可替換成(2+2)*(2+2)=16?
2.2 帶參數的宏定義出現的問題?
在帶參數的宏定義的使用中,極易引起誤解。例如我們需要做個宏替換能求任何數的平方,這就需要使用參數,以便在程序中用實際參數來替換宏定義中的參數。一般學生容易寫成如下形式:?
#define?? area(x)?? x*x?
這在使用中是很容易出現問題的,看如下的程序?
void main()?
{?
int?? y=area(2+2);?
printf(“%d”,y);?
??? }?
按理說給的參數是2+2,所得的結果應該為4*4=16,但是錯了,因為該程序的實際結果為8,仍然是沒能遵循純粹的簡單替換的規則,又是先計算再替換了,在這道程序里,2+2即為area宏中的參數,應該由它來替換宏定義中的x,即替換成2+2*2+2=8了。那如果遵循(1)中的解決辦法,把2+2 括起來,即把宏體中的x括起來,是否可以呢?#define?? area(x) (x)*(x),對于area(2+2),替換為(2+2)*(2+2)=16,可以解決,但是對于area(2+2)/area(2+2)又會怎么樣呢,有的學生一看到這道題馬上給出結果,因為分子分母一樣,又錯了,還是忘了遵循先替換再計算的規則了,這道題替換后會變為 (2+2)*(2+2)/(2+2)*(2+2)即4*4/4*4按照乘除運算規則,結果為16/4*4=4*4=16,那應該怎么呢?解決方法是在整個宏體上再加一個括號,即#define?? area(x) ((x)*(x)),不要覺得這沒必要,沒有它,是不行的。?
要想能夠真正使用好宏定義,那么在讀別人的程序時,一定要記住先將程序中對宏的使用全部替換成它所代表的字符串,不要自作主張地添加任何其他符號,完全展開后再進行相應的計算,就不會寫錯運行結果。如果是自己編程使用宏替換,則在使用簡單宏定義時,當字符串中不只一個符號時,加上括號表現出優先級,如果是帶參數的宏定義,則要給宏體中的每個參數加上括號,并在整個宏體上再加一個括號。看到這里,不禁要問,用宏定義這么麻煩,這么容易出錯,可不可以摒棄它,那讓我們來看一下在C語言中用宏定義的好處吧。?

3?? 宏定義的優點?
(1)?? 方便程序的修改?
使用簡單宏定義可用宏代替一個在程序中經常使用的常量,這樣在將該常量改變時,不用對整個程序進行修改,只修改宏定義的字符串即可,而且當常量比較長時,我們可以用較短的有意義的標識符來寫程序,這樣更方便一些。我們所說的常量改變不是在程序運行期間改變,而是在編程期間的修改,舉一個大家比較熟悉的例子,圓周率π是在數學上常用的一個值,有時我們會用3.14來表示,有時也會用3.1415926等,這要看計算所需要的精度,如果我們編制的一個程序中要多次使用它,那么需要確定一個數值,在本次運行中不改變,但也許后來發現程序所表現的精度有變化,需要改變它的值,這就需要修改程序中所有的相關數值,這會給我們帶來一定的不便,但如果使用宏定義,使用一個標識符來代替,則在修改時只修改宏定義即可,還可以減少輸入 3.1415926這樣長的數值多次的情況,我們可以如此定義 #define?? pi?? 3.1415926,既減少了輸入又便于修改,何樂而不為呢??
(2) 提高程序的運行效率?
使用帶參數的宏定義可完成函數調用的功能,又能減少系統開銷,提高運行效率。正如C語言中所講,函數的使用可以使程序更加模塊化,便于組織,而且可重復利用,但在發生函數調用時,需要保留調用函數的現場,以便子函數執行結束后能返回繼續執行,同樣在子函數執行完后要恢復調用函數的現場,這都需要一定的時間,如果子函數執行的操作比較多,這種轉換時間開銷可以忽略,但如果子函數完成的功能比較少,甚至于只完成一點操作,如一個乘法語句的操作,則這部分轉換開銷就相對較大了,但使用帶參數的宏定義就不會出現這個問題,因為它是在預處理階段即進行了宏展開,在執行時不需要轉換,即在當地執行。宏定義可完成簡單的操作,但復雜的操作還是要由函數調用來完成,而且宏定義所占用的目標代碼空間相對較大。所以在使用時要依據具體情況來決定是否使用宏定義。?

4 結語?
本文對C語言中宏定義#define在使用時容易出現的問題進行了解析,并從C源程序處理過程的角度對#define的處理進行了分析,也對它的優點進行了闡述。只要能夠理解宏展開的規則,掌握使用宏定義時,是在預處理階段對源程序進行替換,只是用對應的字符串替換程序中出現的宏名,這樣就可在正確使用的基礎上充分享受使用宏定義帶來的方便和效率了?

二.?
最近看com相關的資料,看到CCmdTarget實現com接口的時候,去讀了一些宏的定義,在afxdisp.h頭文件中?

#define BEGIN_INTERFACE_PART(localClass, baseClass) /?
class X##localClass : public baseClass /?

本來這個宏定義很容易理解的,但是這里多出個X##,我真沒見過這種用法,不曉得它是什么用意。?
后來問了幾個朋友也都不知道。?

你知道么??

也許你也不知道~呵呵,最后我還是找到了相關的資料,解讀了這個define,還順便認識了另外兩個不常用的define?

#define Conn(x,y) x##y?
#define ToChar(x) #@x?
#define ToString(x) #x?

x##y表示什么?表示x連接y,舉例說:?
int n = Conn(123,456);?? 結果就是n=123456;?
char* str = Conn("asdf", "adf")結果就是 str = "asdfadf";?
怎么樣,很神奇吧?

再來看#@x,其實就是給x加上單引號,結果返回是一個const char。舉例說:?
char a = ToChar(1);結果就是a='1';?
做個越界試驗char a = ToChar(123);結果是a='3';?
但是如果你的參數超過四個字符,編譯器就給給你報錯了!error C2015: too many characters in constant?? :P?

最后看看#x,估計你也明白了,他是給x加雙引號?
char* str = ToString(123132);就成了str="123132";?

最后留幾個小試驗給大家去測測:?
#define Dec(x,y) (x-y)?
int n = Dec( A(123,1), 1230);?
n = Conn(123, Conn(123,332) );?
char* str = A("12", ToString( Dec(3,1));?
結果會如何呢? 嘿嘿嘿嘿~?



define的多行定義?

define可以替代多行的代碼,例如MFC中的宏定義(非常的經典,雖然讓人看了惡心)?

#define MACRO(arg1, arg2) do { /?
/* declarations */ /?
stmt1;?? /?
stmt2;?? /?
/* ... */? /?
} while(0) /* (no trailing ; ) */?
關鍵是要在每一個換行的時候加上一個"/" 至此,一個基本的define框架基本完成,我們可以解讀大部分的define代碼了。我們也可以說,define不過就是一個簡單的代碼替換的一種體制而已,沒有什么神秘的東西。?

三.?
#define xxx() {}?
標準C支持的?
#define xxx() ({})?
GCC新增的功能,主要為了防止宏展開出現問題,默認展開時是要加上一個;的,容易出問題。?


CODE:?
#define A(a,b,c) ({a=1;b+=1;c=3;a+b+c;})?
#include <stdio.h>?
int main()?
{?
?????? int a;?
?????? int b=1;?
?????? int c;?
?????? int d;?
?????? d=A(a,b,c);?
?????? printf("%d,%d,%d,%d/n",a,b,c,d);?
?????? return 0;?
}?
表示該宏函數還有返回值,最后一個式子的返回值作為宏函數的返回值。?
運行結果:?
1,2,3,6

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/388534.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/388534.shtml
英文地址,請注明出處:http://en.pswp.cn/news/388534.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

rabbitmq channel參數詳解【轉】

1、Channel 1.1 channel.exchangeDeclare()&#xff1a; type&#xff1a;有direct、fanout、topic三種durable&#xff1a;true、false true&#xff1a;服務器重啟會保留下來Exchange。警告&#xff1a;僅設置此選項&#xff0c;不代表消息持久化。即不保證重啟后消息還在。原…

感染EXE文件代碼(C++)

C代碼#include <windows.h> #include <winnt.h> #include <stdio.h> #include <assert.h> #define DEBUG 1 #define EXTRA_CODE_LENGTH 18 #define SECTION_SIZE 0x1000 #define SECTION_NAME ".eViLhsU" #define F…

nlp gpt論文_GPT-3:NLP鎮的最新動態

nlp gpt論文什么是GPT-3&#xff1f; (What is GPT-3?) The launch of Open AI’s 3rd generation of the pre-trained language model, GPT-3 (Generative Pre-training Transformer) has got the data science fraternity buzzing with excitement!Open AI的第三代預訓練語言…

真實不裝| 阿里巴巴新人上路指北

新手上路&#xff0c;總想聽聽前輩們分享他們走過的路。橙子選取了阿里巴巴合伙人逍遙子&#xff08;阿里巴巴集團CEO&#xff09; 、Eric&#xff08;螞蟻金服董事長兼CEO&#xff09;、Judy&#xff08;阿里巴巴集團CPO&#xff09;的幾段分享&#xff0c;他們是如何看待職場…

小程序學習總結

上個周末抽空了解了一下小程序,現在將所學所感記錄以便日后翻看;需要指出的是我就粗略過了下小程序的api了解了下小程序的開發流程以及工具的使用,然后寫了一個小程序的demo;在我看來,如果有前端基礎學習小程序無異于錦上添花了,而我這個三年的碼農雖也寫過不少前端代碼但離專業…

tomcat java環境配置

jsp 環境變量配置 一、配置JDK 首先&#xff0c;從Sun網站上下載jdk。 雙擊jdk-1_5_0_04-windows-i586-p.exe開始安裝&#xff0c;默認安裝到C:/Program Files/Java/jdk1.5.0_04&#xff0c;你也可以更改路徑&#xff0c;但要記住最后選擇的路徑&#xff0c;設置環境變量的時候…

uber 數據可視化_使用R探索您在Uber上的活動:如何分析和可視化您的個人數據歷史記錄

uber 數據可視化Perhaps, dear reader, you are too young to remember that before, the only way to request a particular transport service such as a taxi was to raise a hand to make a signal to an available driver, who upon seeing you would stop if he was not …

java B2B2C springmvc mybatis電子商城系統(四)Ribbon

2019獨角獸企業重金招聘Python工程師標準>>> 一&#xff1a;Ribbon是什么&#xff1f; Ribbon是Netflix發布的開源項目&#xff0c;主要功能是提供客戶端的軟件負載均衡算法&#xff0c;將Netflix的中間層服務連接在一起。Ribbon客戶端組件提供一系列完善的配置項如…

c語言函數的形參有幾個,C中子函數最多有幾個形參

該樓層疑似違規已被系統折疊 隱藏此樓查看此樓C89 31個&#xff0c;C99 127個。ANSI C892.2.4.1 Translation limitsThe implementation shall be able to translate and execute at least one program that contains at least one instance of every one of the following lim…

Linux上Libevent的安裝

1、下載wget -O libevent-2.0.21-stable.tar.gz https://github.com/downloads/libevent/libevent/libevent-2.0.21-stable.tar.gz2、解壓 tar zxvf libevent-2.0.21-stable.tar.gz3、配置安裝路徑 cd libevent-2.0.21-stable ./configure -prefix/usr4、編譯并安裝 make make …

Win7安裝oracle 10 g

開始-運行-輸入hdwwiz-回車 ——選則手動 ——網絡適配器——左邊選Microsoft&#xff0c;右邊找到Microsoft Loopback Adapter ——完成 打開 控制面板\網絡和 Internet\網絡和共享中心 會發現多了一個本地連接 點詳細信息 發現是Microsoft Loopback Adapter的。…

基于plotly數據可視化_[Plotly + Datashader]可視化大型地理空間數據集

基于plotly數據可視化簡介(我們將創建的內容)&#xff1a; (Introduction (what we’ll create):) Unlike the previous tutorials in this map-based visualization series, we will be dealing with a very large dataset in this tutorial (about 2GB of lat, lon coordinat…

Centos用戶和用戶組管理

inux系統是一個多用戶多任務的分時操作系統&#xff0c;任何一個要使用系統資源的用戶&#xff0c;都必須首先向系統管理員申請一個賬號&#xff0c;然后以這個賬號的身份進入系統。1、添加新的用戶賬號使用useradd命令&#xff0c;其語法如下&#xff1a;useradd 選項 用戶名-…

吹氣球問題的C語言編程,C語言怎樣給一個數組中的數從大到小排序

滿意答案#include "stdio.h"int main(){int i,j;int a[12];for(i1; i<10; i)scanf("%d",&a[i]);for(i1; i<10; i)for(ji; j<10; j)if(a[i]{int ta[i];a[i]a[j];a[j]t;}//前十個數的排序for(i1; i<10; i)printf("%d ",a[i]);prin…

裴波那契數列

斐波那契數列&#xff1a;0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, ... 求斐波那契數列第 n 項的值&#xff1a; 方法一&#xff1a;遞歸 function fibonacci(n) {if (!Number.isSafeInteger(n) || n < 0) {return;}if (n 0 || n 1) {return n;} else {return fibo…

劃痕實驗 遷移面積自動統計_從Jupyter遷移到合作實驗室

劃痕實驗 遷移面積自動統計If you want to use Google Colaboratory to perform your data analysis, for building data pipelines and data visualizations, here is the beginners’ guide to migrate from one tool to the other.如果您想使用Google Colaboratory進行數據分…

英法德三門語言同時達到c1,【分享】插翅而飛的孩子(轉載)

微信轉來的&#xff0c;覺得發人深思&#xff0c;轉來這里插翅而飛的孩子(一)開篇一&#xff1a;讓孩子擁有一雙豐滿的翅膀。作者簡介&#xff1a;英華蘭的Dr.Bing,德國兒童教育學博士&#xff0c;數字媒體碩士和計算機軟件工程本科。精通英法德三門語言&#xff0c;從事兒童語…

數據庫建表賦予權限語句

sqlplus /nologconn / as sysdba//創建臨時表空間create temporary tablespace zfmi_temptempfile D:\oracle\oradata\zfmi\zfmi_temp.dbf size 32m autoextend on next 32m maxsize 2048mextent management local;//tempfile參數必須有//創建數據表空間create tablespace zfmi…

day03 基本數據類型

1.什么是數據類型 變量值即我們 存放的數據 &#xff0c;數據類型及變量值的類型 2.變量值為何要區分類型 因為變量值使用記錄現實世界中事物的特征&#xff0c;針對不同的特征就應該用不同類型的值去標識 3.如何應用數據類型 一 數據類型&#xff1a; 1.整型int &#xff1a;…

美國移民局的I797表原件和I129表是什么呢

I-129表,Petition for a Non-immigrant Worker&#xff0c;即非移民工作許可申請表I797 表 &#xff0c;Original L1-1A approval notice L1簽證批準通過通知表L-1簽證的申請程序1. L-1簽證的申請必須首先由準備調派雇員的外國母公司在美國的分支機構向移民局提出陳情申請。這些…