c语言typedef作用,C语言typedef的使用

news/2025/2/25 22:51:00

C语言typedef的使用

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;

上面定义了一个名为sorter的函数指针来作为一个新的类型名称,接着使用它来定义名为quicksort和bubblesort的函数指针。

下面总结一下 typedef 的用途:

1、与#define的区别

typedef 行为有点像 #define 宏,用其实际类型替代同义字。不同点是 typedef 在编译时被解释,因此让编译器来应付超越预处理器能力的文本替换。

2、减少错误

定义一种类型的别名,而不只是简单的.宏替换。可以用作同时声明指针型的多个对象。比如:

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 char * pstr;

int mystrcmp(pstr, pstr);

这里将带我们到达第一个 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。


http://www.niftyadmin.cn/n/4390358.html

相关文章

mysql时间戳14小时_SpringBoot时间戳与MySql数据库记录相差14小时排错

项目中遇到存储的时间戳与真实时间相差14小时的现象,以下为解决步骤.问题CREATE TABLE incident (id int(11) NOT NULL AUTO_INCREMENT,created_time timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,recovery_time timestamp NULL DEFAULT NULL,PRIMARY KEY (id)) ENGINEInnoDB…

sql插入日期_SQL入门基础-了解SQL

通过上述的四个步骤,来进入sql的学习。什么是数据库?什么是SQL?数据库是用来存储数据的工具。那和excel有什么区别呢?excel像一个移动硬盘,而数据库像一个网盘,可供多人在上面操作。我们接下来聊的数据库主要是关系数…

输出英文字母的后继字母 c语言,Java 【打印俄文的英文字母】

俄文的的字符可以用 A 到 Я 。public class main {public static void main(String args[]) {char S А, C Я;System.out.println("俄文字母共有:" ((int) C - (int) S 1) "个");for (char i S; i < C; i) {System.out.print(i " ")…

python requests 发起http POST 请求

python requests 发起http POST 请求&#xff0c;带参数&#xff0c;带请求头&#xff1a; #!/usr/bin/env python # -*- coding: utf-8 -*-import requests import jsonurl http://official-account/app/messages/group body {"type": "text", "co…

aprioir算法_DW-DM实验(李向东)Clementine7(关联规则9章).doc

Clementine 关联规则【流9(9.2).str】9.2.3 Apriori算法的应用示例这里&#xff0c;利用Clementine提供的超市顾客个人信息和他们的一次购买商品数据&#xff0c;讲解Aprioir算法的具体操作。数据文件名为BASKETS.txt&#xff0c;为文本格式文件。数据包括两大部分的内容&#…

java script和java有什么区别_Java和JavaScript有什么区别?

程序员和用户经常在Java和JavaScript之间感到困惑。认为Java和JavaScript相同。如果从表面看&#xff0c;它们可能看起来一样。 由于Java和JavaScript均指相同的OOP语言&#xff0c;并且共享相同的控制结构和运算符。但是&#xff0c;当您单独学习它们时&#xff0c;您将了解它…

k8s mysql 持久化_K8s——MySQL实现数据持久化

1、搭建nfs存储[rootdocker-k8s01 ~]# yum -y install nfs-utils[rootdocker-k8s01 ~]# mkdir /nfsdata/mysql -p[rootdocker-k8s01 ~]# cat /etc/exports/nfsdata *(rw,sync,no_root_squash)[rootdocker-k8s01 ~]# systemctl restart nfs-server[rootdocker-k8s01 ~]# systemc…

java导出csv文件_【干货分享】DM数据库使用UTL_FILE读写文件方法

UTL_FILE包可以用来读写操作系统上的文件&#xff0c;提供了在客户端操作服务器端文件的功能。它提供一套严格的使用标准操作系统文件I/O方式&#xff1a;OPEN、 PUT、 GET和 CLOSE操作&#xff1b;其中&#xff0c;GET方法用于读文件&#xff0c;PUT方法用于写文件。当用户读取…