不卡AV在线|网页在线观看无码高清|亚洲国产亚洲国产|国产伦精品一区二区三区免费视频

學(xué)習(xí)啦>學(xué)習(xí)英語>專業(yè)英語>計算機英語>

c中define的用法

時間: 長思709 分享

  c中define的用法的用法你知道嗎?下面小編就跟你們詳細(xì)介紹下c中define的用法的用法,希望對你們有用。

  c中define的用法的用法如下:

  1、define--(無參數(shù)宏定義)用法

  一般形式為:#define

  標(biāo)識符 字符串

 ?。?)"#"表示其為預(yù)處理命令,凡是以"#"開頭的都是預(yù)處理命令;"define"為宏定義命令;"標(biāo)識符"為所定義的宏名;"字符串"可以是常數(shù)、表達(dá)式、字符串等。

 ?。?)除了經(jīng)常用到的如"#define MAXNUM 100"之外,還有其它多種靈活的用法,如"#define M (y*y+3*y)",注意,的表達(dá)式中()是必須的,否則在進行如"2*M+2"的運算

  時就會出現(xiàn)錯誤。

 ?。?)末尾不需要分號(如果加分號,會連同分號一起代換)。

 ?。?)define只是做簡單的文本替換。

 ?。?)作用域為宏定義開始,到源程序結(jié)束,終止定義域可用"#undef M".

 ?。?)宏名如果以字符串的形式被調(diào)用,則不做替換,如printf("I M O")。

 ?。?)可以嵌套進行定義,如

  #define PI 3.14

  #define S PI*R*R

 ?。?)習(xí)慣上把宏名寫為大寫的,以區(qū)別普通的變量。

  2、define和typedef的區(qū)別

  define宏定義是在預(yù)處理完成的,typedef實在編譯時處理的,typedef不是簡單的代換,而是對類型說明符的重命名。

  例如:#define P1 int*

  typedef int* P2;

  P1 a, b;//相當(dāng)于int* a, b,此時a是int指針,b是int整數(shù)。

  P2 a, b;//表示a和b都是int指針。

  3、define(帶參數(shù)宏定義)用法

  一般形式為:#define 宏名(形參) 字符串

  最長見的 #define MAX(a,b) (a>b)?a:b

 ?。?)宏名和形參之間不能有空格。如果上式寫為 #define MAX (a,b) (a>b)?a:b,則MAX就表示整個后面的部分了。

 ?。?)帶參宏定義的形參是不分配內(nèi)存的。

 ?。?) 在宏定義中的形參是標(biāo)識符,而宏調(diào)用中的實參可以是表達(dá)式。

  #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.程序第七行宏調(diào)用中實參為a+1,是一個表達(dá)式,在宏展開時,用a+1代換y,再用(y)*(y) 代換SQ,得到如下語句 :

  sq=(a+1)*(a+1);

  這與函數(shù)的調(diào)用是不同的,函數(shù)調(diào)用時要把實參表達(dá)式的值求出來再賦予形參。而宏代換中對實參表達(dá)式不作計算直接地照原樣代換。

 ?。?) 在宏定義中,字符串內(nèi)的形參通常要用括號括起來以避免出錯。在上例中的宏定義中(y)*(y)表達(dá)式的y都用括號括起來,因此結(jié)果是正確的。如果去掉括號,把程序改 為以下形式:

  #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);

  }

  運行結(jié)果為:

  input a number:3

  sq=7

 ?。?)define的多行定義

  define可以替代多行的代碼,例如MFC中的宏定義

  #define MACRO(arg1, arg2) do { /

  /* declarations */ /

  stmt1; /

  stmt2; /

  /* … */ /

  } while(0) /* (no trailing ; ) */

  關(guān)鍵是要在每一個換行的時候加上一個"/"

  4、define宏和函數(shù)的區(qū)別

  (1)宏定義可以幫助我們防止出錯,提高代碼的可移植性和可讀性等。

  看一個例子,比較兩個數(shù)或者表達(dá)式大小,首先我們把它寫成宏定義:

  #define MAX( a, b) ( (a) > (b) (a) : (b) )

  其次,把它用函數(shù)來實現(xiàn):

  int max( int a, int b)

  {

  return (a > b a : b)

  }

  很顯然,我們不會選擇用函數(shù)來完成這個任務(wù),原因有兩個:

  首先,函數(shù)調(diào)用會帶來額外的開銷,它需要開辟一片??臻g,記錄返回地址,將形參壓棧,從函數(shù)返回還要釋放堆棧。這種開銷不僅會降低代碼效率,

  而且代碼量也會大大增加,而使用宏定義則在代碼規(guī)模和速度方面都比函數(shù)更勝一籌;其次,函數(shù)的參數(shù)必須被聲明為一種特定的類型,所以它只能在

  類型合適的表達(dá)式上使用,我們?nèi)绻容^兩個浮點型的大小,就不得不再寫一個專門針對浮點型的比較函數(shù)。反之,上面的那個宏定義可以用于整

  形、長整形、單浮點型、雙浮點型以及其他任何可以用">"操作符比較值大小的類型,也就是說,宏是與類型無關(guān)的。

  和使用函數(shù)相比,使用宏的不利之處在于每次使用宏時,一份宏定義代碼的拷貝都會插入到程序中。除非宏非常短,否則使用宏會大幅度增加程序的長

  度。

  還有一些任務(wù)根本無法用函數(shù)實現(xiàn),但是用宏定義卻很好實現(xiàn)。比如參數(shù)類型沒法作為參數(shù)傳遞給函數(shù),但是可以把參數(shù)類型傳遞給帶參的宏。

  看下面的例子:

  #define MALLOC(n, type) /

 ?。?(type *) malloc((n)* sizeof(type)))

  利用這個宏,我們就可以為任何類型分配一段我們指定的空間大小,并返回指向這段空間的指針。我們可以觀察一下這個宏確切的工作過程:

  int *ptr;

  ptr = MALLOC ( 5, int );

  將這宏展開以后的結(jié)果:

  ptr = (int *) malloc ( (5) * sizeof(int) );

  這個例子是宏定義的經(jīng)典應(yīng)用之一,完成了函數(shù)不能完成的功能,但是宏定義也不能濫用,通常,如果相同的代碼需要出現(xiàn)在程序的幾個地方,更好的

  方法是把它實現(xiàn)為一個函數(shù)。

 ?。?)下面總結(jié)和宏和函數(shù)的不同之處,以供大家寫代碼時使用,這段總結(jié)摘自《C和指針》一書。

  代碼長度

  #define宏:每次使用時,宏代碼都被插入到程序中。除了非常小的宏之外,程序的長度將大幅度增長

  函數(shù):函數(shù)代碼只出現(xiàn)于一個地方:每次使用這個函數(shù)時,都調(diào)用那個地方的同一份代碼

  執(zhí)行速度

  #define宏:更快

  函數(shù): 存在函數(shù)調(diào)用、返回的額外開銷

  操作符優(yōu)先級

  #define宏:宏參數(shù)的求值是在所有周圍表達(dá)式的上下文環(huán)境里,除非它們加上括號,否則鄰近操作符的優(yōu)先級可能產(chǎn)生不可預(yù)料的結(jié)果。

  函數(shù):函數(shù)參數(shù)只在函數(shù)調(diào)用時求值一次,它的結(jié)果值傳遞給函數(shù)。表達(dá)式的求值結(jié)果更容易預(yù)測。

  參數(shù)求值

  #define宏:參數(shù)用于宏定義時,每次都將重新求值,由于多次求值,具有副作用的參數(shù)可能會產(chǎn)生不可預(yù)測的結(jié)果。

  函數(shù):參數(shù)在函數(shù)調(diào)用前只求值一次,在函數(shù)中多次使用參數(shù)并不會導(dǎo)致多次求值過程,參數(shù)的副作用并不會造成任何特殊問題。

  參數(shù)類型

  #define宏:宏與類型無關(guān),只要參數(shù)的操作是合法的,它可以用于任何參數(shù)類型。

  函數(shù): 函數(shù)的參數(shù)是與類型有關(guān)系的,如果參數(shù)的類型不同,就需要使用不同的函數(shù),即使它們執(zhí)行的任務(wù)是相同的。

537133