(六)C语言之typedef详解
1、typedef可以看作type define的缩写,顾名思义就是类型定义,也就是说它只是给已有的类型重新定义了一个方便使用的别名,并没有产生新的数据类型。typedef的使用与宏定义define有些许的相似,但两者又有以下不同:
1.1.与#define不同,typedef给出的符号名称仅限于对类型,而不是对值。
1.2.typedef的解释由编译器,而不是预处理器执行。
1.3.typedef比#define更灵活。
2、既然typedef没有定义新的数据类型,那么为什么还要使用它呢?使用typedef有其自身的优势:
2.1.它使得定义更加直观,从定义就可了解变量的某些信息。如
typedef unsigned int BYTE;
BYTE x, y[10], *z;
2.2.它可以使程序参数化,以提高程序的可移植性。如
ime_t time(time_t *); 该函数返回的是time_t 类型的返回值,有些系统中 time_t 被定义为unsigned long类型,而另外一些系统中,可能被定义为unsigned int 类型,这样,在移植到不同的系统中时,只要改变typedef定义,就可以在不同的系统中进行移植了。
2.3.表达方式更加简洁。如
使用typedef命名一个结构体时,typedef struct{double x; double y;}rect; rect r1 = {3.0, 6.0};
如果不使用typedef则显得复杂,struct {double x; double y;}r1 = {3.0, 6.0};
使用typedef定义的作用域取决于typedef语句所在的位置,如果定义是在一个函数内部,它的作用域就是局部的,限定在那个函数里。如果定义是在函数外部,它将具有全局作用域。typedef中声明的类型在变量名的位置出现,而不是紧接在关键字typedef之后。typedef在语法上类似于存储类中的extern、static等,所以不能同时对一个变量类型使用typedef和static等。建立好数据类型名之后,可以使用它来进行类型声明、类型转换等。如:
typedef char *String;
String p, lineptr[MAXLINES], alloc(int); //类型声明
int strcmp(String, String);
p = (String)malloc(100); //类型转换
3、typedef的常用范例如下:
3.1.简单的定义变量的别名。
typedef char * PChar;
PChar a, b; //相当于char *a; char *b;
3.2.与结构体的结合使用。
typedef struct Node
{
int a;
char *b;
}*PNode;
PNode a, b;
3.3.typedef引入函数指针。
下面是参考别的博客的内容:
typedef int (*PF) (const char *, const char *);这个声明引入了 PF 类型作为函数指针的同义字,该函数有两个 const char * 类型的参数以及一个 int 类型的返回值。对复杂变量建立一个类型别名的方法很简单,你只要在传统的变量声明表达式里用类型名替代变量名,然后把关键字typedef加在该语句的开头就行了。比如:
void (*signal (int signr,void (*handler)(int))) (int);可以通过两次typedef来进行定义。
typedef void sigfunc(int);
sigfunc *signal(int signr,sigfunc *handler);
其中typedef定义了一个有一个整型参数无返回值的函数类型。void (*handler)(int)表示一个有一个整型参数无返回值的函数指针,这个指针名为handler,所以其可以用sigfunc进行说明,此时sigfunc就相当于前面的int signr中int的作用;同理这个函数也是这样。
理解复杂声明可用的“右左法则”:从变量名看起,先往右,再往左,碰到一个圆括号就调转阅读的方向;括号内分析完就跳出括号,还是按先右后左的顺序,如此循环,直到整个声明分析完。举例:
int (*func)(int *p);
首先找到变量名func,外面有一对圆括号,而且左边是一个*号,这说明func是一个指针;然后跳出这个圆括号,先看右边,又遇到圆括号,这说明(*func)是一个函数,所以func是一个指向这类函数的指针,即函数指针,这类函数具有int*类型的形参,返回值类型是int。
int (*func[5])(int *);
func右边是一个[]运算符,说明func是具有5个元素的数组;func的左边有一个*,说明func的元素是指针(注意这里的*不是修饰func,而是修饰func[5]的,原因是[]运算符优先级比*高,func先跟[]结合)。跳出这个括号,看右边,又遇到圆括号,说明func数组的元素是函数类型的指针,它指向的函数具有int*类型的形参,返回值类型为int。