多应用+插件架构,代码干净,二开方便,首家独创一键云编译技术,文档视频完善,免费商用码云13.8K 广告
在实际问题中,一组数据往往具有不同的数据类型。例如,在学生登记表中,姓名为字符型,学号为整型或字符型,年龄为整型,性别为字符型,成绩为整型或实数型别,显然,不能使用一个数组来存放这一组数据。因为数组中各个元素的类型和长度都必须一致,以便于编译系统处理。为了解决这个问题,C语言中给出了另一种构造数据类型–结构。他相当于其他高级语言中的记录。 “结构”是一种构造类型,他是由若干“成员”组成的。每一个成员可以是一个基本数据类型或者又是一个构造类型。结构是一种“构造”而成的数据类型,在说明和使用之前先定义它,也就是构造它。就像是定义函数一样。 1:结构的定义 定义一个结构的一般形式为: struct 结构名 { 成员列表 }; 2:结构类型变量的说明 说明结构变量有3中方法,以结构stu为例。 ~~~ struct stu { int num; char name[20]; char sex; float score; }; //说明结构变量 struct stu boy1,boy2; ~~~ 说明变量boy1和boy2为stu结构类型。也可以使用宏定义用一个符号常量来表示一个结构类型,例如 ~~~ #define STU struct stu STU { int num; char name[20]; char sex; float score; }; STU boy1,boy2; ~~~ 还可以在定义结构类型的同时说明结构变量,例如: ~~~ struct stu { int num; char name[20]; char sex; float score; } boy1,boy2; ~~~ 或者是直接说明结构变量,例如 ~~~ struct { int num; char name[20]; char sex; float score; }boy1,boy2; ~~~ 注意: 如果结构变量是全局变量或为静态变量,则可以对它进行 初始化赋值。对局部或自动结构变量不能进行初始化赋值。 3:结构数组 继续使用上面stu的例子: struct stu boys[20]; 4:结构指针变量 当使用一个指针变量指向一个结构变量时,称之为结构指针变量。结构变量中的值是所指向的结构变量的首地址。通过结构指针即可访问该结构 变量,这与数组指针和函数指针的情况是相同的。结构指针变量说明的一般形式为: struct 结构名 *结构指针变量名 例如,如果要说明一个指向stu的指针变量pstu,可写为: struct stu *pstu; 当然,也可以在定义stu结构的时候同时说明pstu。与前面变量的声明相同。结构指针变量必须要先赋值后才能使用。赋值是把结构变量的首地址赋予该指针变量,不能把结构名赋予该指针变量,如过boy是被说明为stu类型的结构变量,则”pstu=&boy”是正确的,但是”pstu=&stu”是错误的。 结构名和结构变量是两个不同的概念,不能混淆。结构名只能表示一个结构形式,编译系统并不对他分配空间。只有当变量说明为这种类型的结构时,才对该变量分配内存空间。因此”&stu”的写法是错误的,不可能去取一个结构名的首地址。有了结构指针变量,就能更方便的访问结构变量的各个成员。 程序访问的一般形式为:(*结构指针变量).成员名 或为: 结构指针变量->成员名 例如”(*pstu).num”或者”pstu->num”。 应该注意(*pstu)两侧的括号不可少,因为成员符号“.”的优先级高于”*“。如果去掉括号协作*pstu.num,就等效于”*(pstu.num)”,这样意义就不对了。 5:结构指针变量作为函数参数 在ANSI C标准中允许使用结构变量作为函数参数进行整体传送。但是这种传送要将全部成员逐个传送,特别是成员为数组时将会使传送的时间和空间开销很大,严重的降低了程序的效率。因此最好的办法就是使用指针,即用指针变量作为函数参数进行传送,由于实参传向星灿的只是地址,从而减少了时间和空间的开销。 在本例中,假设一个通讯录由以下几项数据信息组成 数据项 类型 姓名 字符串 地址 字符串 邮政编码 字符串 电话号码 字符串 下面请看一下代码的实现: ~~~ #include <stdio.h> #include <stdlib.h> #include <string.h> #define ZIPLEN 10 #define PHONLEN 10 struct addr{ char *name; /* 姓名 */ char *address; /* 地址 */ char zip[ZIPLEN]; /* 邮编 */ char phone[PHONLEN]; /* 手机号码 */ }; int readaddr(struct addr *dpt); int writeaddr(struct addr *dpt); /** * 编制一个包含姓名,地址,邮编和电话的通讯录 * 输入输出函数。 */ int main() { struct addr p[1]; int i,j; for(i=0;readaddr(p+i);i++); for(j = 0;j < i;j++) writeaddr(p+j); puts("Press any key to quit...\n"); getch(); return 0; } /** * 向结构中的成员赋值 * */ int readaddr(struct addr *dpt) { int len; char buf[120]; //输入字符串的缓冲区 printf("Please input the Name:\n"); if(scanf("%s",buf)==1){ len = strlen(buf); dpt->name = (char *)malloc(len+1); strcpy(dpt->name,buf); }else return 0; printf("Please input the Address:\n"); if(scanf("%s",buf) == 1){ len = strlen(buf); dpt->address = (char *)malloc(len+1); strcpy(dpt->address,buf); }else{ free(dpt->name); return 0; } printf("Please input the Zip:\n"); if(scanf("%s",buf) == 1){ strncpy(dpt->zip,buf,ZIPLEN-1); }else{ free(dpt->name); free(dpt->address); return 0; } printf("Please input the Phone:\n"); if(scanf("%s",buf) == 1){ strncpy(dpt->phone,buf,PHONLEN-1); }else{ free(dpt->name); free(dpt->address); return 0; } return 1; } /** * 输出通讯录 */ int writeaddr(struct addr *dpt) { printf("Name : %s\n",dpt->name); /* 输出姓名 */ printf("Address : %s\n",dpt->address); /* 输出地址 */ printf("Zip : %s\n",dpt->zip); /* 输出邮编 */ printf("Phone : %s\n",dpt->phone); /* 输出手机号 */ } ~~~ 下面是运行结果: ![这里写图片描述](https://box.kancloud.cn/2016-05-24_5743c076269dc.jpg "") 注意:关于动态内存分配问题: 在数组中,数组的长度是预先定义好的,在整个程序中固定不变。C语言不允许动态数组类型。例如”int n;scanf(“%d”,n);int a[n];”,这是错误的。但是在实际编译过程中,往往会发生这种情况,即所需的内存空间取决于实际输入的数据,而无法预先确定。对于这种问题,用数组的办法很难解决。为了解决上述问题,C语言提供了一些内存管理函数,这些内存管理函数可以按照需要动态分配内存空间,也可以把不再使用的空间收回待用。常用的内存管理函数有3个: 1:分配内存空间函数malloc。调用形式 (类型说明符 *)malloc(size); 在内存的动态存储区中分配一块长度为”size”字节的连续区域。函数的返回值为该区域的首地址。 2:分配内存空间函数 calloc。calloc也用于分配内存空间,调用形式: (类型说明符 *)calloc(n,size); 在内存动态存储区中分配n快长度为”size”字节的连续区域。函数的返回值为该区域的首地址。 3:释放内存空间函数free.调用形式 free(void *ptr); 释放ptr所指向的一块内存区域,ptr是一个任意类型的指针变量,他指向被释放区域的首地址。被释放区域应是由malloc或calloc函数所分配的区域。