工学1号馆

home

C语言 typedef 的使用

Wu Yudong    December 02, 2016     C   1,063   

typedef 关键字能帮助你简化复杂的定义并让你的代码简洁可靠,当然,可靠这一点我还是持保留态度,因人而异吧。

C 语言提供 typedef 关键字,允许你为已经存在的类型起一个新的名字,有一点需要注意,typedef 仅仅是为已经存在的类型创建了一个新的名字而已,不是创建新的类型。

首先将 typedef 关键字放在前面,接着是已经存在的类型名,接着是新的名字:

typedef existing_type new_name;

举个例子,如果你想为 unsinged integer 创建一个新的名叫 score 的类型,你仅需要使用这样使用 typedef :

typedef unsigned int score;

后面你就可以使用“新类型” score 来声明变量,例如:

score high_score;

当然你还可以同时定义多个变量

typedef int aaa, bbb, ccc;
typedef int ar[15], arr[9][6];
typedef char c, *cp, carr[100];

/* 下面定义一些对象 */
/* 都是int */
aaa     int1;
bbb     int2;
ccc     int3;

ar      yyy;    /* 容量为15的整型数组 */
arr     xxx;    /* 9*6 阶的整型数组 */

c       ch;     /* 一个字符 */
cp      pnt;    /* 指向字符的指针 */
carr    chry;   /* 容量为100的字符数组 */

C 语言 typedef 的使用

这里来看一下使用typedef的好处,假如有一天你想将 unsigned int 修改成更大的数据类型,比如unsigned long,而此时你的很多代码文件都使用了 unsigned int,如果没有使用typedef,你得到处都修改一遍,十分麻烦。如果使用了之前的定义,这时只需修改score一处就行,是不是很方便?

另一个使用 typedef 来增强可靠性的例子是 size_t,size_t 是 sizeof 操作符指定类型的大小,其基于目标处理器的运算能力而不是内存能力。只要你的程序包含stddef.h库,你将获得平台上的任何对象的准确大小。

接下来举一些typedef 在函数指针、结构体、枚举等中的应用

使用 typedef 作用于结构体:

为了定义一个复杂的数据,定义一个如下的结构体:

struct complex {
    float real;
    float imag;
}; 
struct complex a, b;

如果不使用typedef, 你必须在每一个变量声明的地方使用 struct 关键字,然而,如果你使用了 tpedef 定义 complex 类型的数,你只需要使用complex number, you can omit the struct keyword whenever you declare a new variable. 因此使用typedef可以帮助你简化变量的定义。

typedef struct {
    float real;
    float imag;
} complex;
complex a, b;

使用 typedef 作用于联合体:

假如你将用户名和密码来登录,这时定义一个account 结构体,其中包含 account_name 联合体:

typedef union {
    char *username;
    char *email;
} account_name;

typedef struct {
    account_name name;
    char *password;
} account;
account user1, user2;

使用 typedef 作用于枚举:

下面的例子描述了怎样使用 typedef 定义一个枚举:

typedef enum { red, green, blue } RGB ;
RGB color;

如果你不使用 typedef ,每次定义变量的时候都要加上 enum 关键字:

enum RGB { red, green, blue };
enum RGB color;

使用 typedef 作用于函数指针:

一个函数指针是指向一个函数的指针,你可以使用 typedef 来给一个函数指针取一个简单的名字,看下面的例子:

typedef int (*sorter)(void* a, size_t size);
sorter quicksort, bubblesort;
char* pa, pb; // 它只声明了一个指向字符变量的指针,和一个字符变量;

下面的符合我们的预期

typedef char* PCHAR;  
PCHAR pa, pb;

这种用法很有用,特别是char* pa, pb的定义,初学者往往认为是定义了两个字符型指针,其实不是,而用typedef char* PCHAR就不会出现这样的问题,减少了错误的发生。

3、平台无关性

用typedef来定义与平台无关的类型。用 typedef 来定义机器无关的类型,例如,你可以定义一个叫 REAL 的浮点类型,在目标机器上它可以获得最高的精度:

typedef long double REAL;

在不支持 long double 的机器上,该 typedef 看起来会是下面这样:

typedef double REAL;

也就是说,当跨平台时,只要改下 typedef 本身就行,不用对其他源码做任何修改。

标准库就广泛使用了这个技巧,比如size_t。另外,因为typedef是定义了一种类型的新别名,不是简单的字符串替换,所以它比宏来得稳健。

4、掩饰复合类型

typedef 还可以掩饰复合类型,如指针和数组。

例如,你不用像下面这样重复定义有 81 个字符元素的数组:

char line1[81];  
char line2[81];

定义一个 typedef,每当要用到相同类型和大小的数组时,可以这样:

typedef char Line[81];

同样,可以象下面这样隐藏指针语法:

这里将带我们到达第一个 typedef 陷阱。标准函数 strcmp()有两个‘ const char *’类型的参数。因此,它可能会误导人们象下面这样声明 mystrcmp():

int mystrcmp(const pstr, const pstr);

用GNU的gcc和g++编译器,是会出现警告的,按照顺序,‘const pstr’被解释为‘char* const‘(一个指向 char 的指针常量),两者表达的并非同一意思。为了得到正确的类型,应当如下声明:

typedef const char* pstr;

5、代码简化

代码简化。为复杂的声明定义一个新的简单的别名。方法是:在原来的声明里逐步用别名替换一部分复杂声明,如此循环,把带变量名的部分留到最后替换,得到的就是原声明的最简化版。举例:

原声明:

void (*b[10]) (void (*)());

变量名为b,先替换右边部分括号里的,pFunParam为别名

typedef void (*pFunParam)();

再替换左边的变量b,pFunx为别名二:

typedef void (*pFunx)(pFunParam);

原声明的最简化版:

pFunx b[10];

理解复杂声明可用的“右左法则”:从变量名看起,先往右,再往左,碰到一个圆括号就调转阅读的方向;括号内分析完就跳出括号,还是按先右后左的顺序,如此循环,直到整个声明分析完。举例:

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。

如果文章对您有帮助,欢迎点击下方按钮打赏作者

Comments

总结得非常全面详细,学习了


Cancel reply
To verify that you are human, please fill in "七"(required)