當前位置:首頁 » 編程語言 » c語言undef
擴展閱讀
webinf下怎麼引入js 2023-08-31 21:54:13
堡壘機怎麼打開web 2023-08-31 21:54:11

c語言undef

發布時間: 2022-08-29 18:14:42

c語言的 #undef 問題

i = MIN + 5; 正確的。因為你前面 有 #define MIN 30
printf("%d\n", MIN); 這里 沒有定義,因為你前面一句 #undef MIN 取消了 MIN 的定義。

② c語言的# undef是什麼意思

終止宏定義命令!
比如
#define N 10
int main(void)
{
.....
.....
....
}
#undef N
這個N的值有效范圍就只有在主函數中有效

③ C語言中#define的問題

你的理解是不對的,
#ifdef
global_variables_here
#undef
extern
#define
extern
#endif
應該這樣理解:
如果定義了global_variables_here
不管extern是否定義過,都取消他之前的定義
重新定義extern
這種做法的目的在於防止extern重定義,比如你在另外一個文件中定義過了extern的話,你不做#undef的操作,直接在現在這個文件中#define
extern是錯誤的做法,因為編譯器不知道該用哪個宏定義。

④ C中預定義標識符define include undef ifdef ifndef endif line 的用法以及意義最好帶例子

由預處理程序自動完成的。在C語言中,「宏」分為有參數和無參數兩種。 下面分別討論這兩種「宏」的定義和調用。

無參宏定義
無參宏的宏名後不帶參數。其定義的一般形式為: #define 標識符 字元串 其中的「#」表示這是一條預處理命令。凡是以「#」開頭的均為預處理命令。「define」為宏定義命令。 「標識符」為所定義的宏名。「字元串」可以是常數、表達式、格式串等。在前面介紹過的符號常量的定義就是一種無參宏定義。 此外,常對程序中反復使用的表達式進行宏定義。例如: # define M (y*y+3*y) 定義M表達式(y*y+3*y)。在編寫源程序時,所有的(y*y+3*y)都可由M代替,而對源程序作編譯時,將先由預處理程序進行宏代換,即用(y*y+3*y)表達式去置換所有的宏名M,然後再進行編譯。
#define M (y*y+3*y)
main(){
int s,y;
printf("input a number: ");
scanf("%d",&y);
s=3*M+4*M+5*M;
printf("s=%d\n",s);
}
上常式序中首先進行宏定義,定義M表達式(y*y+3*y),在s= 3*M+4*M+5* M中作了宏調用。在預處理時經宏展開後該語句變為:s=3*(y*y+3*y)+4(y*y+3*y)+5(y*y+3*y);但要注意的是,在宏定義中表達式(y*y+3*y)兩邊的括弧不能少。否則會發生錯誤。
當作以下定義後: #difine M y*y+3*y在宏展開時將得到下述語句: s=3*y*y+3*y+4*y*y+3*y+5*y*y+3*y;這相當於; 3y�2+3y+4y�2+3y+5y�2+3y;顯然與原題意要求不符。計算結果當然是錯誤的。 因此在作宏定義時必須十分注意。應保證在宏代換之後不發生錯誤。對於宏定義還要說明以下幾點:

1. 宏定義是用宏名來表示一個字元串,在宏展開時又以該字元串取代宏名,這只是一種簡單的代換,字元串中可以含任何字元,可以是常數,也可以是表達式,預處理程序對它不作任何檢查。如有錯誤,只能在編譯已被宏展開後的源程序時發現。

2. 宏定義不是說明或語句,在行末不必加分號,如加上分號則連分號也一起置換。

3. 宏定義必須寫在函數之外,其作用域為宏定義命令起到源程序結 束。如要終止其作用域可使用# undef命令,例如: # define PI 3.14159
main()
{
……
}
# undef PIPI的作用域
f1()
....表示PI只在main函數中有效,在f1中無效。
4. 宏名在源程序中若用引號括起來,則預處理程序不對其作宏代換。
#define OK 100
main()
{
printf("OK");
printf("\n");
}
上例中定義宏名OK表示100,但在printf語句中OK被引號括起來,因此不作宏代換。程序的運行結果為:OK這表示把「OK」當字元串處理。

5. 宏定義允許嵌套,在宏定義的字元串中可以使用已經定義的宏名。在宏展開時由預處理程序層層代換。例如: #define PI 3.1415926
#define S PI*y*y /* PI是已定義的宏名*/對語句: printf("%f",s);在宏代換後變為: printf("%f",3.1415926*y*y);

6. 習慣上宏名用大寫字母表示,以便於與變數區別。但也允許用小寫字母。

7. 可用宏定義表示數據類型,使書寫方便。例如: #define STU struct stu在程序中可用STU作變數說明: STU body[5],*p;#define INTEGER int 在程序中即可用INTEGER作整型變數說明: INTEGER a,b; 應注意用宏定義表示數據類型和用typedef定義數據說明符的區別。宏定義只是簡單的字元串代換,是在預處理完成的,而typedef是在編譯時處理的,它不是作簡單的代換, 而是對類型說明符重新命名。被命名的標識符具有類型定義說明的功能。請看下面的例子: #define PIN1 int* typedef (int*) PIN2;從形式上看這兩者相似, 但在實際使用中卻不相同。下面用PIN1,PIN2說明變數時就可以看出它們的區別: PIN1 a,b;在宏代換後變成 int *a,b;表示a是指向整型的指針變數,而b是整型變數。然而:PIN2 a,b;表示a,b都是指向整型的指針變數。因為PIN2是一個類型說明符。由這個例子可見,宏定義雖然也可表示數據類型, 但畢竟是作字元
代換。在使用時要分外小心,以避出錯。

8. 對「輸出格式」作宏定義,可以減少書寫麻煩。例9.3 中就採用了這種方法。
#define P printf
#define D "%d\n"
#define F "%f\n"
main(){
int a=5, c=8, e=11;
float b=3.8, d=9.7, f=21.08;
P(D F,a,b);
P(D F,c,d);
P(D F,e,f);
}

帶參宏定義

C語言允許宏帶有參數。在宏定義中的參數稱為形式參數, 在宏調用中的參數稱為實際參數。對帶參數的宏,在調用中,不僅要宏展開, 而且要用實參去代換形參。

帶參宏定義的一般形式為: #define 宏名(形參表) 字元串 在字元串中含有各個形參。帶參宏調用的一般形式為: 宏名(實參表);
例如:
#define M(y) y*y+3*y /*宏定義*/
:
k=M(5); /*宏調用*/
: 在宏調用時,用實參5去代替形參y, 經預處理宏展開後的語句
為: k=5*5+3*5
#define MAX(a,b) (a>b)?a:b
main(){
int x,y,max;
printf("input two numbers: ");
scanf("%d%d",&x,&y);
max=MAX(x,y);
printf("max=%d\n",max);
}
上常式序的第一行進行帶參宏定義,用宏名MAX表示條件表達式(a>b)?a:b,形參a,b均出現在條件表達式中。程序第七行max=MAX(x,
y)為宏調用,實參x,y,將代換形參a,b。宏展開後該語句為: max=(x>y)?x:y;用於計算x,y中的大數。對於帶參的宏定義有以下問題需要說明:

1. 帶參宏定義中,宏名和形參表之間不能有空格出現。
例如把: #define MAX(a,b) (a>b)?a:b寫為: #define MAX (a,b) (a>b)?a:b 將被認為是無參宏定義,宏名MAX代表字元串 (a,b)(a>b)?a:b。
宏展開時,宏調用語句: max=MAX(x,y);將變為: max=(a,b)(a>b)?a:b(x,y);這顯然是錯誤的。

2. 在帶參宏定義中,形式參數不分配內存單元,因此不必作類型定義。而宏調用中的實參有具體的值。要用它們去代換形參,因此必須作類型說明。這是與函數中的情況不同的。在函數中,形參和實參是兩個不同的量,各有自己的作用域,調用時要把實參值賦予形參,進行「值傳遞」。而在帶參宏中,只是符號代換,不存在值傳遞的問題。

3. 在宏定義中的形參是標識符,而宏調用中的實參可以是表達式。
#define SQ(y) (y)*(y)
main(){
int a,sq;
printf("input a number: ");
scanf("%d",&a);
sq=SQ(a+1);
printf("sq=%d\n",sq);
}
上例中第一行為宏定義,形參為y。程序第七行宏調用中實參為a+1,是一個表達式,在宏展開時,用a+1代換y,再用(y)*(y) 代換SQ,得到如下語句: sq=(a+1)*(a+1); 這與函數的調用是不同的, 函數調用時要把實參表達式的值求出來再賦予形參。 而宏代換中對實參表達式不作計算直接地照原樣代換。

4. 在宏定義中,字元串內的形參通常要用括弧括起來以避免出錯。 在上例中的宏定義中(y)*(y)表達式的y都用括弧括起來,因此結果是正確的。如果去掉括弧,把程序改為以下形式:
#define SQ(y) y*y
main(){
int a,sq;
printf("input a number: ");
scanf("%d",&a);
sq=SQ(a+1);
printf("sq=%d\n",sq);
}
運行結果為:input a number:3
sq=7 同樣輸入3,但結果卻是不一樣的。問題在哪裡呢? 這是由於代換只作符號代換而不作其它處理而造成的。 宏代換後將得到以下語句: sq=a+1*a+1; 由於a為3故sq的值為7。這顯然與題意相違,因此參數兩邊的括弧是不能少的。即使在參數兩邊加括弧還是不夠的,請看下面程序:
#define SQ(y) (y)*(y)
main(){
int a,sq;
printf("input a number: ");
scanf("%d",&a);
sq=160/SQ(a+1);
printf("sq=%d\n",sq);
}
本程序與前例相比,只把宏調用語句改為: sq=160/SQ(a+1); 運行本程序如輸入值仍為3時,希望結果為10。但實際運行的結果如下:input a number:3 sq=160為什麼會得這樣的結果呢?分析宏調用語句,在宏代換之後變為: sq=160/(a+1)*(a+1);a為3時,由於「/」和「*」運算符優先順序和結合性相同, 則先作160/(3+1)得40,再作40*(3+1)最後得160。為了得到正確答案應在宏定義中的整個字元串外加括弧, 程序修改如下
#define SQ(y) ((y)*(y))
main(){
int a,sq;
printf("input a number: ");
scanf("%d",&a);
sq=160/SQ(a+1);
printf("sq=%d\n",sq);
}
以上討論說明,對於宏定義不僅應在參數兩側加括弧, 也應在整個字元串外加括弧。

5. 帶參的宏和帶參函數很相似,但有本質上的不同,除上面已談到的各點外,把同一表達式用函數處理與用宏處理兩者的結果有可能是不同的。main(){
int i=1;
while(i<=5)
printf("%d\n",SQ(i++));
}
SQ(int y)
{
return((y)*(y));
}#define SQ(y) ((y)*(y))
main(){
int i=1;
while(i<=5)
printf("%d\n",SQ(i++));
}
在上例中函數名為SQ,形參為Y,函數體表達式為((y)*(y))。在例9.6中宏名為SQ,形參也為y,字元串表達式為(y)*(y))。 兩例是相同的。例9.6的函數調用為SQ(i++),例9.7的宏調用為SQ(i++),實參也是相同的。從輸出結果來看,卻大不相同。分析如下:在例9.6中,函數調用是把實參i值傳給形參y後自增1。 然後輸出函數值。因而要循環5次。輸出1~5的平方值。而在例9.7中宏調用時,只作代換。SQ(i++)被代換為((i++)*(i++))。在第一次循環時,由於i等於1,其計算過程為:表達式中前一個i初值為1,然後i自增1變為2,因此表達式中第2個i初值為2,兩相乘的結果也為2,然後i值再自增1,得3。在第二次循環時,i值已有初值為3,因此表達式中前一個i為3,後一個i為4, 乘積為12,然後i再自增1變為5。進入第三次循環,由於i 值已為5,所以這將是最後一次循環。計算表達式的值為5*6等於30。i值再自增1變為6,不再滿足循環條件,停止循環。從以上分析可以看出函數調用和宏調用二者在形式上相似, 在本質上是完全不同的。

6. 宏定義也可用來定義多個語句,在宏調用時,把這些語句又代換到源程序內。看下面的例子。
#define SSSV(s1,s2,s3,v) s1=l*w;s2=l*h;s3=w*h;v=w*l*h;
main(){
int l=3,w=4,h=5,sa,sb,sc,vv;
SSSV(sa,sb,sc,vv);
printf("sa=%d\nsb=%d\nsc=%d\nvv=%d\n",sa,sb,sc,vv);
}
程序第一行為宏定義,用宏名SSSV表示4個賦值語句,4 個形參分別為4個賦值符左部的變數。在宏調用時,把4 個語句展開並用實參代替形參。使計算結果送入實參之中。

文件包含

文件包含是C預處理程序的另一個重要功能。文件包含命令行的一般形式為: #include"文件名" 在前面我們已多次用此命令包含過庫函數的頭文件。例如:
#include"stdio.h"
#include"math.h"
文件包含命令的功能是把指定的文件插入該命令行位置取代該命令行, 從而把指定的文件和當前的源程序文件連成一個源文件。在程序設計中,文件包含是很有用的。 一個大的程序可以分為多個模塊,由多個程序員分別編程。 有些公用的符號常量或宏定義等可單獨組成一個文件, 在其它文件的開頭用包含命令包含該文件即可使用。這樣,可避免在每個文件開頭都去書寫那些公用量, 從而節省時間,並減少出錯。

對文件包含命令還要說明以下幾點:
1. 包含命令中的文件名可以用雙引號括起來,也可以用尖括弧括起來。例如以下寫法都是允許的: #include"stdio.h" #include<math.h> 但是這兩種形式是有區別的:使用尖括弧表示在包含文件目錄中去查找(包含目錄是由用戶在設置環境時設置的), 而不在源文件目錄去查找; 使用雙引號則表示首先在當前的源文件目錄中查找,若未找到才到包含目錄中去查找。 用戶編程時可根據自己文件所在的目錄來選擇某一種命令形式。

2. 一個include命令只能指定一個被包含文件, 若有多個文件要包含,則需用多個include命令。3. 文件包含允許嵌套,即在一個被包含的文件中又可以包含另一個文件。

條件編譯

預處理程序提供了條件編譯的功能。 可以按不同的條件去編譯不同的程序部分,因而產生不同的目標代碼文件。 這對於程序的移植和調試是很有用的。 條件編譯有三種形式,下面分別介紹:
1. 第一種形式:
#ifdef 標識符
程序段1
#else
程序段2
#endif
它的功能是,如果標識符已被 #define命令定義過則對程序段1進行編譯;否則對程序段2進行編譯。如果沒有程序段2(它為空),本格式中的#else可以沒有, 即可以寫為:
#ifdef 標識符
程序段 #endif
#define NUM ok
main(){
struct stu
{
int num;
char *name;
char sex;
float score;
} *ps;
ps=(struct stu*)malloc(sizeof(struct stu));
ps->num=102;
ps->name="Zhang ping";
ps->sex='M';
ps->score=62.5;
#ifdef NUM
printf("Number=%d\nScore=%f\n",ps->num,ps->score);
#else
printf("Name=%s\nSex=%c\n",ps->name,ps->sex);
#endif
free(ps);
}
由於在程序的第16行插入了條件編譯預處理命令, 因此要根據NUM是否被定義過來決定編譯那一個printf語句。而在程序的第一行已對NUM作過宏定義,因此應對第一個printf語句作編譯故運行結果是輸出了學號和成績。在程序的第一行宏定義中,定義NUM表示字元串OK,其實也可以為任何字元串,甚至不給出任何字元串,寫為: #define NUM 也具有同樣的意義。 只有取消程序的第一行才會去編譯第二個printf語句。讀者可上機試作。

2. 第二種形式:
#ifndef 標識符
程序段1
#else
程序段2
#endif
與第一種形式的區別是將「ifdef」改為「ifndef」。它的功能是,如果標識符未被#define命令定義過則對程序段1進行編譯, 否則對程序段2進行編譯。這與第一種形式的功能正相反。

3. 第三種形式:
#if 常量表達式
程序段1
#else
程序段2
#endif
它的功能是,如常量表達式的值為真(非0),則對程序段1 進行編譯,否則對程序段2進行編譯。因此可以使程序在不同條件下,完成不同的功能
#define R 1
main(){
float c,r,s;
printf ("input a number: ");
scanf("%f",&c);
#if R
r=3.14159*c*c;
printf("area of round is: %f\n",r);
#else
s=c*c;
printf("area of square is: %f\n",s);
#endif
}
本例中採用了第三種形式的條件編譯。在程序第一行宏定義中,定義R為1,因此在條件編譯時,常量表達式的值為真, 故計算並輸出圓面積。上面介紹的條件編譯當然也可以用條件語句來實現。 但是用條件語句將會對整個源程序進行編譯,生成的目標代碼程序很長,而採用條件編譯,則根據條件只編譯其中的程序段1或程序段2, 生成的目標程序較短。如果條件選擇的程序段很長, 採用條件編譯的方法是十分必要的。

本章小結
1. 預處理功能是C語言特有的功能,它是在對源程序正式編譯前由預處理程序完成的。程序員在程序中用預處理命令來調用這些功能。

2. 宏定義是用一個標識符來表示一個字元串,這個字元串可以是常量、變數或表達式。在宏調用中將用該字元串代換宏名。

3. 宏定義可以帶有參數,宏調用時是以實參代換形參。而不是「值傳送」。

4. 為了避免宏代換時發生錯誤,宏定義中的字元串應加括弧,字元串中出現的形式參數兩邊也應加括弧。

5. 文件包含是預處理的一個重要功能,它可用來把多個源文件連接成一個源文件進行編譯,結果將生成一個目標文件。

6. 條件編譯允許只編譯源程序中滿足條件的程序段,使生成的目標程序較短,從而減少了內存的開銷並提高了程序的效率。

7. 使用預處理功能便於程序的修改、閱讀、移植和調試,也便於實現模塊化程序設計

⑤ c語言define用法是什麼

1、簡單的define定義#define MAXTIME 1000

一個簡單的MAXTIME就定義好了,它代表1000,如果在程序裡面寫if(i<MAXTIME){.........}

編譯器在處理這個代碼之前會對MAXTIME進行處理替換為1000。

這樣的定義看起來類似於普通的常量定義CONST,但也有著不同,因為define的定義更像是簡單的文本替換,而不是作為一個量來使用,這個問題在下面反映的尤為突出。

2、define的「函數定義」

define可以像函數那樣接受一些參數,#define max(x,y) (x)>(y)?(x):(y);

這個定義就將返回兩個數中較大的那個,看到了嗎?因為這個「函數」沒有類型檢查,就好像一個函數模板似的,當然,它絕對沒有模板那麼安全就是了。可以作為一個簡單的模板來使用而已。

C語言中無參宏定義的一般形式:

#define 宏名 字元串,其中的「#」表示這是一條預處理命令。凡是以「#」開頭的均為預處理命令。「define」為宏定義命令。「標識符」為所定義的宏名。「字元串」可以是常數、表達式、格式串等。

例如:#define M (a+b)它的作用是指定標識符M來代替表達式(a+b)。在編寫源程序時,所有的(a+b)都可由M代替,而對源程序作編譯時,將先由預處理程序進行宏代換,即用(a+b)表達式去置換所有的宏名M,然後再進行編譯。

⑥ C語言中define的問題

你的理解是不對的,
#ifdef GLOBAL_VARIABLES_HERE
#undef EXTERN
#define EXTERN
#endif

應該這樣理解:
如果定義了GLOBAL_VARIABLES_HERE
不管EXTERN是否定義過,都取消他之前的定義
重新定義EXTERN

這種做法的目的在於防止EXTERN重定義,比如你在另外一個文件中定義過了EXTERN的話,你不做#undef的操作,直接在現在這個文件中#define EXTERN是錯誤的做法,因為編譯器不知道該用哪個宏定義。

⑦ 關於C語言的一個簡單問題

這句話是不對的,宏定義可以位於文件頭,函數外,甚至函數中,比如以下代碼:

分別在頭部定義了紅 A_VALUE,main函數中定義了宏 B_VALUE,在main函數之後又定義了C_VALUE 宏,可以看到,都是可以的,其中,#undef 是取消宏定義,這里不需要關心。

⑧ c語言define用法

1、#ifdef:

如果有定義的#ifdef的一般形式為:#ifdef 宏替換名 語句段 #endif

含義為:如果宏替換名已被定義過,則對「語句段」進行編譯;如果未定義#ifdef後面的宏替換名,則不對語句段進行編譯。


2、#ifndef:

如果無定義的#ifndef的一般形式為:#ifndef 宏替換名 語句段 #endif

含義為:如果未定義#ifndef後面的宏替換名,則對「語句段1」進行編譯;如果定義#ifndef後面的宏替換名,則不執行語句段。


3、#undef:

#undef命令可以刪除事先定義了的宏定義。

#undef命令的一般形式如下:#undef 宏替換名

(8)c語言undef擴展閱讀:

c語言中define作為英文單詞的含義:

define是一個英語單詞,動詞、名詞,作動詞時意為「 定義;使明確;規定」,作名詞時意為 「(Define)人名;(英)德法恩;(葡)德菲內」。

短語搭配:

flow define流程定義

Define Mask定義罩框 ; 界說罩框

define byte[計]定義位元組 ; 定義位元組指令 ; 偽指令 ; 位元組定義

⑨ C語言,#define語句的用法。

1.簡單的宏定義

格式:#define 標識符 替換列表

說明:替換列表可以是數,字元串字面量,標點符號,運算符,標識符,關鍵字,字元常量。

注意:替換列表是可以為空的

典型錯誤:

#defineN=100
inta[N];/*這樣會成為inta[=100],這里會處理成為一種標識記號一樣*/


#defineN100;
inta[N];/*帶分號的定義會成為inta[100;],這是一種很常見的錯誤*/

#definepin(int*);
pina,b;
int*a,b;/*本意是a和b都是int型指針,但是實際上變成int*a,b;a是int型指針,而b是int型變數。這是應該使用typedef來代替define,這樣a和b就都是int型指針了。*/

典型的使用方法:

使用宏定義可以自己根據自己的習慣來定義甚至改變C語言的語法習慣,例如:

#defineBEGIN{
#defineEND}
intmain()BEGIN
printf("DEFINE---- ");
END

定義一個循環

#defineLOOPfor(;;)

重新定義數據類型

#defineITint

2.帶參數的宏

#define 標識符(x1,x2,x3...xn) 替換列表 (注意:x1,x2,x3..是宏的參數,標識符和其左括弧之間不能有空格)

使用方法:

#defineMAX(x,y)((x)>(y)?(x):(y))
i=MAX(j+k,m-n);

替換為:

i=MAX((j+k)>(m-n)?(j+k):(m-n));
#defineSI_EX(n)((n)%2==0)
if(SI_EX(i))i++;

替換為:

if(SI_EX((i)%2==0))

通過以上例子可以看出,標識符帶參數(X1,X2,X3.....)在替換時會被替換列表(Y1,Y2,Y3....)對應的替換,但是和順序無關。

3.宏的特殊單行定義

#defineA(x)T_##x
#defineB(x)#@x
#defineC(x)#x
x=1

替換為:

A(1)------> T_1 (T_##x,##符號是記號粘黏符號,將前後的字元粘黏起來。)

B(1)------> '1' ( #@x , #@ 符號會將宏的參數進行字元串字面量化,並且加『』號)

C(1)------> "1" ( #x ,#符號會將宏的參數進行字元串字面量化,並且加""號)

4.define的多行定義

define可以替代多行的代碼,在每一個換行的時候加上一個""

#defineMAX(X,Y)do{
語句1;
語句2;
/*注釋的寫法*/
}while(0)/*(notrailing;)*/

5.在大規模的開發過程中,特別是跨平台和系統的軟體里,define最重要的功能是條件編譯。

#ifdefWINDOWS
......
......
#endif
#ifdefLINUX
......
......
#endif

可以在編譯的時候通過#define設置編譯環境

6.取消宏

#undef標識符

7.條件編譯

#ifdefXXX…(#else)…#endif


8.預定義宏

在C語言中收錄了一些有用的宏,這些宏是提供當前編譯信息的。

__LINE__ 被編譯文件的行數

(整型)

__FILE__ 被編譯文件的名字 (字元型)

__DATE__ 編譯日期 (字元型)

__TIME__ 編譯時間 (字元型)

__STDC__ 如果編譯器接受標准C,那麼值為1. (整型)

通常在編譯的時候可以用這些預定義的宏來進行排錯,例如:

一般情況下,當出現除零的情況時編譯器是沒有提示的,可以自定義一個解決辦法。

#defineCHECKZERO(divisor)
if(divisor==0)
printf("***ATTEMPTTODIVIDEBYZEROINLINE%doffile%s*** ",__LINE__,__FILE__);

引用方法

CHECKZERO(j);
k=i/j;