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=%d\n",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=%d\n",max); } 上例程序的第一行进行带参宏定义,用宏名MAX表示条件表达式 (a>b)?a:b , 形参a,b均出现在条件表达式中。程序中 max=MAX(x,y) 为宏调用,实参x,y,将代换形参a,b。宏展开后该语句为: max=(x>y)?x:y; 用于计算printf是什么意思c++x,y中的大数。 #define 条件编译 头文件(.h)可以被头文件或C文件包含;重复包含(重复定义)由于头文件包含可以嵌套,那么C文件就有可能包含多次同一个头文件,就可能出现重复定义的问题的。 通过条件编译开关来避免重复包含(重复定义) 例如 #ifndef __headerfileXXX__ #define __headerfileXXX__ 文件内容 #endif
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论