亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

C語言初學者入門講座 第十五講 預處理

2019-11-17 05:43:33
字體:
來源:轉載
供稿:網友

  概述

  在前面各章中,已多次使用過以“#”號開頭的預處理命令。如包含命令# include,宏定義命令# define等。在源程序中這些命令都放在函數之外, 而且一般都放在源文件的前面,它們稱為預處理部分。

  所謂預處理是指在進行編譯的第一遍掃描(詞法掃描和語法分析)之前所作的工作。預處理是C語言的一個重要功能, 它由預處理程序負責完成。當對一個源文件進行編譯時, 系統將自動引用預處理程序對源程序中的預處理部分作處理, 處理完畢自動進入對源程序的編譯。

  C語言提供了多種預處理功能,如宏定義、文件包含、 條件編譯等。合理地使用預處理功能編寫的程序便于閱讀、修改、 移植和調試,也有利于模塊化程序設計。本章介紹常用的幾種預處理功能。

  宏定義

  在C語言源程序中答應用一個標識符來表示一個字符串, 稱為“宏”。被定義為“宏”的標識符稱為“宏名”。在編譯預處理時,對程序中所有出現的“宏名”,都用宏定義中的字符串去代換, 這稱為“宏代換”或“宏展開”。

  宏定義是由源程序中的宏定義命令完成的。 宏代換是由預處理程序自動完成的。在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, 生成的目標程序較短。假如條件選擇的程序段很長, 采用條件編譯的方法是十分必要的。



發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品成人一区二区三区吃奶| 美女久久久久久久久久久| 久久精品久久精品亚洲人| 欧美激情免费视频| 国产综合色香蕉精品| 亚洲天堂av网| 亚洲国产精品久久久久秋霞蜜臀| 成人黄色免费网站在线观看| 亚洲精品免费在线视频| 日本高清+成人网在线观看| 日韩欧美在线播放| 亚洲精品国产精品国自产观看浪潮| 国产精品中文久久久久久久| 欧美日韩久久久久| 91精品视频免费看| 亚洲天堂日韩电影| 欧美高清在线播放| 欧美一级高清免费播放| 国产丝袜一区二区三区免费视频| 亚洲国产精品嫩草影院久久| 日韩在线观看高清| 国产精品99蜜臀久久不卡二区| 国产亚洲成av人片在线观看桃| 一区二区欧美在线| 国产精品视频男人的天堂| 在线视频免费一区二区| 91最新国产视频| 欧美黄色www| 一区二区在线视频播放| 5566成人精品视频免费| 国产精品久久久久久久久久尿| 欧美极品在线播放| 精品久久久久久中文字幕大豆网| 成人免费网站在线| 91在线视频成人| 欧美在线视频免费观看| 国产xxx69麻豆国语对白| 亚洲美女av黄| 国产主播精品在线| 日本一区二区三区四区视频| 国产精品免费久久久久影院| 欧美日韩国产中文字幕| 亚洲欧美国产制服动漫| 亚洲情综合五月天| 国产精品91久久久| 欧美一区二区色| 久久91超碰青草是什么| 欧美精品www在线观看| 日韩欧美成人区| 伊人久久男人天堂| 精品视频在线观看日韩| 久久精品国产久精国产思思| 亚洲天堂av在线免费观看| 91美女片黄在线观看游戏| 亚洲丝袜在线视频| 精品国产乱码久久久久久天美| 影音先锋欧美在线资源| 精品国产91久久久| 国产精品久久av| 国产精品老女人视频| 欧美性xxxx极品高清hd直播| 97超碰国产精品女人人人爽| 欧美激情一区二区三区久久久| 国产精品视频免费观看www| 欧美最顶级的aⅴ艳星| 日韩黄在线观看| 国产精品91在线| 日韩人体视频一二区| 麻豆国产精品va在线观看不卡| 国产精品啪视频| 亚洲aaaaaa| 午夜精品国产精品大乳美女| 中文字幕亚洲欧美一区二区三区| 国产精品免费久久久久影院| 91免费福利视频| 欧美日韩国产二区| 亚洲第一中文字幕在线观看| 欧美激情一区二区三区在线视频观看| 久久久久久这里只有精品| 日韩欧美高清在线视频| 日韩av手机在线观看| 亚洲aa中文字幕| 久久天天躁狠狠躁夜夜躁2014| 国产成一区二区| 国产成人中文字幕| 91视频国产精品| 欧美午夜精品久久久久久久| 欧美视频裸体精品| 久久婷婷国产麻豆91天堂| 精品国产老师黑色丝袜高跟鞋| 日韩av男人的天堂| 91久久精品久久国产性色也91| 亚洲精品99久久久久中文字幕| 亚洲综合最新在线| 66m—66摸成人免费视频| 欧美日韩中文字幕在线| 国产视频观看一区| 亚洲黄色av女优在线观看| 国产一级揄自揄精品视频| 日韩中文在线视频| 国产精品久久久久久久久久| 亚洲天堂久久av| 在线性视频日韩欧美| 97在线视频精品| 欧美亚洲成人精品| 动漫精品一区二区| 中文字幕亚洲一区在线观看| 欧美裸体xxxx| 欧美在线观看www| 欧美高清自拍一区| 夜夜躁日日躁狠狠久久88av| 中文字幕视频在线免费欧美日韩综合在线看| 欧美激情第99页| 日韩视频在线观看免费| 青青草成人在线| 亚洲欧美国产一本综合首页| 国产偷国产偷亚洲清高网站| 久色乳综合思思在线视频| 黑人狂躁日本妞一区二区三区| 色久欧美在线视频观看| 久久久久久com| 国产精品免费视频久久久| 成人亚洲综合色就1024| 国产精品爽爽爽爽爽爽在线观看| 91在线观看免费观看| 奇米一区二区三区四区久久| 欧洲日韩成人av| 国产精品国产福利国产秒拍| 国产一区二区三区在线| 日本中文字幕成人| 91国产美女在线观看| 亚洲影院色无极综合| 亚洲视频日韩精品| 97视频在线免费观看| 精品久久久久久| 茄子视频成人在线| 成人h片在线播放免费网站| 国产婷婷97碰碰久久人人蜜臀| 黄网站色欧美视频| 日韩精品免费视频| 亚洲视频在线免费观看| 狠狠色狠色综合曰曰| 国产美女被下药99| 亚洲视频在线观看| 久久久精品影院| 色婷婷av一区二区三区久久| 日韩av日韩在线观看| 国产精品成人观看视频国产奇米| 国产精品扒开腿做爽爽爽男男| 一区二区三区美女xx视频| 欧美理论电影在线播放| 欧美午夜宅男影院在线观看| 欧美日韩亚洲视频一区| 奇米4444一区二区三区| 亚洲色图偷窥自拍| 日韩在线免费视频| 97av在线视频免费播放| 国产一区二区三区久久精品| 在线观看免费高清视频97| 2023亚洲男人天堂| 国产精品久久久久久久久久| 日韩av在线网址| 最近2019中文字幕第三页视频| 久久综合久久八八|