admin管理员组

文章数量:1530841

2024年6月13日发(作者:)

typedef与define 比较

1、typedef和define

typedef 在编译时被解释,define在预处理时被替换。

#define定义“可读”的常量以及一些宏语句的任务,而typedef则常用来定义关

键字、冗长的类型的别名。

宏定义只是简单的字符串代换(原地扩展),而typedef则不是原地扩展,它的新

名字具有一定的封装性,以致于新命名的标识符具有更易定义变量的功能。

typedef int size;

#define MAX 100

2、typedef属于存储类声明说明符

typedef int size 是一个完整的声明,不可拆分,不能简单的看作这个语句的意

思是替int起一个别名为size。

3、typedef编译时做的事

typedef int size;

size length;

对size length;这条语句来说,在编译时不是把其中的“size”替换为“int”,真实过

程应该是这样的:

1、length声明为一个size类型。

2、typedef语句后面部分中的语句(即int size)相当于一个一元表达式,其中的

size是一个未知数,现在要用length去代入这个一元表达式得到int length,然后用

这个替换得到的一元表达式取代size length。

3、1和2过程重复,直至一元函数中没有未知数。

4、最终得到int length。

可能从上面这个简单的语句解释不足以说清我的理解思路,我们看一个复杂点

的:

typedef int (*PF) (const char *, const char *);

PF Register(PF pf);

对PF Register(PF pf);这条语句来说,在编译时它的过程应该是这样的:

1、Register(PF pf)声明为PF类型。

2、typedef语句后面部分中的语句(即int (*PF) (const char *, const char *))相当于

一个一元表达式,其中的PF是一个未知数,现在要用Register(PF pf)去代入这个一

元表达式得到int (*Register(PF pf)) (const char *, const char *)),然后用这个替换得到

的一元表达式取代PF Register(PF pf)。

3、1和2过程重复,直至一元函数中没有未知数。

4、最终得到int (*Register(int (*pf) (const char *, const char *))) (const char *,

const char *)。

其中过程3的详细过程:

pf声明为PF类型。

typedef语句后面部分中的语句(即int (*PF) (const char *, const char *))相当于一

个一元表达式,其中的PF是一个未知数,现在要用pf去代入这个一元表达式得到

int (*pf) (const char *, const char *)),然后用这个替换得到的一元表达式取代PF pf。

得到int (*Register(int (*pf) (const char *, const char *))) (const char *, const char *)。

#define 的作用

在C或C++语言源程序中允许用一个标识符来表示一个字符串,称为“宏”。被定义为“宏”

的标识符称为“宏名”。在编译预处理时,对程序中所有出现的“宏名”,都用宏定义中的字符串

去代换,这称为“宏代换”或“宏展开”。宏定义是由源程序中的宏定义命令完成的。宏代换是由

预处理程序自动完成的。在C或C++语言中,“宏”分为有参数和无参数两种。

无参宏定义

无参宏定义的一般形式为: #define 宏名 字符串 无参宏的宏名后不带参

数。 其定义的一般形式为: #define 标识符 字符串 其中的“#”表示这是一条预处理

命令。凡是以“#”开头的均为预处理命令。“define”为宏定义命令。“标识符”为所定义的宏名。

“字符串”可以是常数、表达式、格式串等。 例如: #define M (a+b) 它的作用是指

定标识符M来代替表达式(a+b)。在编写源程序时,所有的(a+b)都可由M代替,而对源程序

作编译时,将先由预处理程序进行宏代换,即用(a+b)表达式去置换所有的宏名M,然后再进行

编译。 程序1: #define M (a+b) int main() { int s,y; printf("input a

number: "); scanf("%d",&y); s=M*M; printf("s=%dn",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: #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=%dn",max); } 上例程序的第一行进行带参宏定义,用宏名MAX表示条

件表达式 (a>b)?a:b , 形参a,b均出现在条件表达式中。程序中 max=MAX(x,y) 为宏调用,

实参x,y,将代换形参a,b。宏展开后该语句为: max=(x>y)?x:y; 用于计算x,y中的大数。

#define 条件编译 头文件(.h)可以被头文件或C文件包含;重复包含(重复定义)由于头

文件包含可以嵌套,那么C文件就有可能包含多次同一个头文件,就可能出现重复定义的问题

的。 通过条件编译开关来避免重复包含(重复定义) 例如 #ifndef __headerfileXXX__

#define __headerfileXXX__ 文件内容 #endif

本文标签: 定义语句表达式预处理字符串