typedef

发布时间 2023-10-17 17:51:40作者: 老嘎皮

typedef 的使用与宏定义 define 有些许的相似,但两者又有以下不同:

1. 与 #define 不同,typedef 给出的符号名称仅限于对类型,而不是对值。

2.typedef 的解释由编译器,而不是预处理器执行。

3.typedef 比 #define 更灵活。

既然 typedef 没有定义新的数据类型,那么为什么还要使用它呢?使用 typedef 有其自身的优势:

1. 它使得定义更加直观,从定义就可了解变量的某些信息。如:

typedef unsigned int UINT;

UINT x, y[10], *z;

 

2. 它可以使程序参数化,以提高程序的可移植性

time_t time (time_t *); 该函数返回的是 time_t 类型的返回值,有些系统中 time_t 被定义为 unsigned long 类型,而另外一些系统中,可能被定义为 unsigned int 类型,这样,在移植到不同的系统中时,只要改变 typedef 定义,就可以在不同的系统中进行移植了。

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); // 类型转换

typedef 的常用范例如下:

1. 简单的定义变量的别名。

typedef char * PChar;

PChar a, b; // 相当于 char *a; char *b;

2. 与结构体的结合使用。

typedef struct Node
{   
int a;   char *b;
}
*PNode;
PNode a, b;

3. 对复杂的变量定义一个类型别名。

下面是参考别的博客的内容:

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。