typedef与define
最近做Linux项目的时候发现自己对typedef认识并不是很深刻,故总结一下,并与看似与define类似做一个比较。
typedef大致会分为以下四个功能:
一、定义一种类型的别名,所以不仅仅只是简单的宏替换。可以用作同时声明指针型的多个对象。
如:如果想定义两个字符变量指针:
一种方法是这样定义:
char *ca, *cb;
还有一种方法是使用typedef去定义:
typedef char* VARIABLE; //这里为了区分一般用大写
VARIABLE ca, cb; //同时声明了两个指向字符变量的指针
两种方法都可以定义两个字符变量指针,但是明显第二种方法代码的鲁棒性会更好。
二、用在旧的C的代码中(具体多旧没有查),帮助struct。在编写一般的代码之中,声明struct新对象时,都要带上struct,即形式为: struct 结构名 对象名。
如:定义一个学生的结构体。
若不使用typedef时:
struct STUDENT
{
char name[20];
int num;
char sex;
int age;
float score;
char addr[30];
};
struct STUDENT stud1, stud2;
使用typedef时:
typedef struct student_data
{
char name[20];
int num;
char sex;
int age;
float score;
char addr[30];
}STUDENT;
STUDENT stud1, stud2;
这样的话少写一个struct,且代码阅读性会变好。
三、用typedef来定义与平台无关的类型。
如:比如定义一个叫 REAL 的浮点类型,在目标平台一上,让它表示最高精度的类型为。
typedef long double REAL;
在不支持 long double 的平台二上,改为:
typedef double REAL;
在连 double 都不支持的平台三上,改为:
typedef float REAL;
这样去定义如果跨平台只需要修改typedef,而不用修改源码。因为typedef是定义了一种类型的新别名,不是简单的字符串替换,所以比宏来得稳健。
四、为复杂的声明定义一个新的简单的别名。
如:
doube(*)() (*e)[9];//原声明
typedef double(*pFuny)();
typedef pFuny (*pFunParamy)[9];
pFunParamy e;//最终版
注意typedef陷阱:
typedef char* PSTR;//定义
int mystrcmp(const PSTR, const PSTR);
/* const PSTR实际上不相当于const char*,它实际上相当于char* const
* 原因在于const给予了整个指针本身以常量性,也就是形成了常量指针char* const
*
* 当const和typedef一起出现时,typedef不会是简单的字符串替换就行
*/
/* typedef在语法上是一个存储类的关键字,
* (如auto、extern、mutable、static、register等一样),
* 虽然它并不真正影响对象的存储特性
*/
typedef static int INT2; //不可行,编译将失败,会提示“指定了一个以上的存储类”
而typedef与define的区别更是天差地别:
一、原理不同
#define是C语言中定义的语法,是预处理指令,在预处理时进行简单而机械的字符串替换,不作正确性检查,只有在编译已被展开的源程序时才会发现可能的错误并报错。
typedef是关键字,在编译时处理,有类型检查功能。它在自己的作用域内给一个已经存在的类型一个别名,但不能在一个函数定义里面使用typedef。用typedef定义数组、指针、结构等类型会带来很大的方便,不仅使程序书写简单,也使意义明确,增强可读性。
二、功能不同
typedef用来定义类型的别名,起到类型易于记忆的功能。另一个功能是定义机器无关的类型。如定义一个REAL的浮点类型,在目标机器上它可以获得最高的精度:typedef long double REAL, 在不支持long double的机器上,看起来是这样的,typedef double REAL,在不支持double的机器上,是这样的,typedef float REAL
#define不只是可以为类型取别名,还可以定义常量、变量、编译开关等。
三、作用域不同
#define没有作用域的限制,只要是之前预定义过的宏,在以后的程序中都可以使用,而typedef有自己的作用域。
四、对指针的操作不同
这个在上面typedef陷阱里已经说过的,详细的请看上面的段落。