1、typedef的用法
简单的例子,如清单1:
- #include <stdio.h>
- typedef int type_int;
- int main(void)
- {
- type_int a = 0x10203040;
- short b = a;
- printf("a = 0x%x, b = 0x%x\n", a, b);
- return 0;
- }
清单1第3行的意思是:为数据类型关键字int声明了一个别名type_int,而且用type_int声明变量的效果与int相同。
例子输出结果:
- a = 0x10203040, b = 0x3040
从结果可知,变量a确实是整型的。
复杂的例子,如清单2:
- #include <stdio.h>
- typedef int *type_int_ptr;
- int main(void)
- {
- type_int_ptr a, b;
- //如果用int *a, b;这样的形式声明变量a和b,其中只有a是指针变量,b是普通的整型变量。
- int c = 3, d = 4;
- a = &c;
- b = &d;
- printf("*a = %d, *b = %d\n", *a, *b);
- return 0;
- }
清单2的第3行的意思是:为数据类型int *声明了一个别名type_int_ptr,而且用type_int_ptr声明变量的效果与int *相同。
例子输出结果:
- *a = 3, *b = 4
从结果可知,变量a、b都为指针。注意与“int *a, b;”声明形式的差异,所以声明指针时星号(*)一定要跟变量名牢牢地结合在一起。
更复杂的例子,如清单3:
- #include <signal.h>
- typedef void (*sighandler_t)(int);
- sighandler_t signal(int signum, sighandler_t handler);
在Linux应用开发中,常会遇到清单3中signal函数的声明形式,其意思是:为综合性的数据类型void (*)(int)声明了一个别名sighandler_t,而且用sighandler_t声明变量的效果与void (*)(int)相同,所以函数signal的返回值和参数handler的数据类型都是void (*)(int)型的函数指针。
2、与#define没有可比性
宏指令#define只是简单的文本替换,而且是在预编译过程中被预编译器处理的。而关键字typedef根本就不是文本替换的概念。
比较过程,参见清单4:
- #include <stdio.h>
- #define myint int
- typedef int type_arr[10];
- int main(void)
- {
- myint a = 3;
- type_arr arr_b;
- arr_b[3] = a;
- printf("a = %d, arr_b[3] = %d\n", a, arr_b[3]);
- return 0;
- }
第3行定义了一个宏myint,替换值为int,在程序预编译后,第8行的“myint a = 3;”语句就被替换为“int a = 3;”,即文本myint被替换为int。于是,不少初学者很快联想到typedef的声明是不是也是类似的过程,如“typedef int type_int; type_int a;”,在这个例子中type_int很像是被int替换了,那么在第4行的typedef声明中你能看出是谁替换了谁?既然不是简单的文本替换,那么应该如何理解typedef声明呢?这里有个诀窍:(1)在typedef声明语句中,如typedef int type_arr[10],去掉关键字typedef(即int type_arr[10]),就可以获知type_arr的数据类型是一个一维数组,并且有10个整型的元素。(2)在用type_arr声明别的变量时,这个被声明的变量的数据类型与上一步获知的数据类型相同。
例子输出结果:
- a = 3, arr_b[3] = 3