#define与typedef的区别

#define与typedef的区别

一.#define 命令是 C 语言中的一个宏定义命令 ,它用来将一个标识符定义为一个字符串 , 该标识符被称为宏名, 被定义的字符串称为替换文本。 

    该命令有两种格式:一种是简单的宏定义, 另一种是带参数的宏定义。

无参宏定义

  C语言中无参宏定义的一般形式为:#define 宏名 字符串

  其中的“#”表示这是一条预处理命令。凡是以“#”开头的均为预处理命令。“define”为宏定义命令。“标识符”为所定义的宏名。“字符串”可以是常数、表达式、格式串等。

  例如: #define M (a+b) 它的作用是指定标识符M来代替表达式(a+b)。在编写源程序时,所有的(a+b)都可由M代替,而对源程序作编译时,将先由预处理程序进行宏代换,即用(a+b)表达式去置换所有的宏名M,然后再进行编译。 

  程序1:
  #include<stdio.h>
  #define M (a+b)
  int main(void)
  {
    int s,a,b;
    printf("inputnumbera&b:");
    scanf("%d%d",&a,&b);
    s = M*M;
    printf("s=%d ",s);
  }

  上例程序中首先进行宏定义,定义M来替代表达式(a+b),在 s= M * M 中作了宏调用。在预处理时经宏展开后该语句变为: S=(a+b)*(a+b) 。但要注意的是,在宏定义中表达式(a+b)两边的括号不能少。否则会发生错误。  如当作以下定义后:#define M (a)+(b)  在宏展开时将得到下述语句:S= (a)+(b)*(a)+(b)。


对于宏定义还要说明以下几点:
1.宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名,这只是一种简单的代换,字符串中可以含任何字符,可以是常数,也可以是表达式,预处理程序对它不作任何检查。如有错误,只能在编译已被宏展开后的源程序时发现。
2.宏定义不是说明或语句,在行末不必加分号,如加上分号则连分号也一起置换。
3.宏定义其作用域为宏定义命令起到源程序结束。如要终止其作用域可使用#undef命令。

有参宏定义

   C语言允许宏带有参数。在宏定义中的参数称为形式参数,在宏调用中的参数称为实际参数。对带参数的宏,在调用中,不仅要宏展开,而且要用实参去代换形参。
带参宏定义的一般形式为:  #define 宏名(形参表)字符串
  在字符串中含有各个形参。 带参宏调用的一般形式为:  宏名(形参表)

  例如:
  #define M(y) ((y)*(y)+3*(y))/*宏定义*/
  k=M(5);/*宏调用*/
  在宏调用时,用实参5去代替形参y,经预处理宏展开后的语句为:  k=5*5+3*5

  程序2:

  #include<stdio.h>

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

  int main(int argc,char *argv[])
  {
      int x,y,max;

      printf("input two numbers: ");
      scanf("%d%d",&x,&y);

      max=MAX(x,y);

      printf("max=%d ",max);
      return 0;
    }
    上例程序的第二行进行带参宏定义,用宏名MAX表示条件表达式 (a>b)?a:b ,形参a,b均出现在条件表达式中。程序中 max=MAX(x,y) 为宏调用,实参x,y,将代换形参a,b。宏展开后该语句为: max=(x>y)?x:y;  用于计算x,y中的大数。

二.typedef在编译的时候执行,用来为复杂的声明定义简单的别名,不能用于字符串,整数等的。

1)typedef的最简单使用
      typedef long byte_4;
      给已知数据类型long起个新名字,叫byte_4。


2) typedef与结构结合使用

      typedef struct tagMyStruct
     {
         int iNum;
         long lLength;
     }MyStruct;

通常讲,typedef要比#define要好,特别是在有指针的场合。

请看例子:
typedef char* pStr1;
#define pStr2 char* 
pStr1 s1,s2;
pStr2 s3,s4;
在上述的变量定义中,s1、s2、s3都被定义为char *,而s4则定义成了char,不是我们所预期的指针变量,根本原因就在于#define只是简单的字符串替换而typedef则是为一个类型起新名字。