c的详细学习(9)结构体与共用体的学习(一)
C语言提供了另外两种构造类型:结构体与公用体,用来存储若干个类型不同但彼此组成一个集合的数据总体。
(1)结构体类型与结构体变量
1.定义
其一般形式为:
struct 结构体类型名{
数据类型1 成员名1;
数据类型2 成员名2;
数据类型3 成员名3;
......
}
定义一个结构体变量只是描述结构体的组织形式,并不意味着将分配一段内存单元来存放个数据项成员。它的作用只是告诉编译系统所定义的结构体类型是有哪些类型的成员构成的,各占多少字节,按什么形式存储,并把他们当作一个整体来处理。
2.结构体变量的定义:
结构体类型的变量的定义有三种方法:
1)先定义结构体类型,再定义结构体变量;struct student{省略成员的定义};struct student stu1,stu2;
2)定义结构体类型的同时,定义结构体变量;struct student{省略成员的定义}stu1,stu2;
3)不定义结构体的类型,直接定义变量。(此时结构体是无名的)struct{省略成员的定义}stu1,stu2;
结构体成员也可以是一个结构体变量,即一个结构体的定义中可以嵌套另外一个结构体的结构体的结构体。
struct date{
int year;
int month;
int day;
};
struct student{
long int number;
char name[20];
char sex;
struct date birthday;
char addr[30];
};
3.结构体变量的引用
在定义结构体类型变量以后,就可以引用结构体类型变量,如赋值,存取和运算等。遵循以下规则:
1)不能将结构体变量当作一个整体处理;
2)访问带有结构体变量成员的结构体时应采取逐级访问的方式;
3)结构体的成员变量和普通变量一样可以进行各种运算。
4.结构体变量的初始化
结构体类型是数组类型的扩充,只是它的成员项可以具有不同的数据类型:
struct student{
long num;
char name[20];
char sex;
char addr[20];
}s1={2006001,"MenAngel",'m',"anhui hefei"};
(2)结构体数组
结构体数组与普通的变量的数组大致相同:
初始化:
struct student{}stu1={{括号里与单个结构体初始化方式相同},{},{}......};
1)输入3名学生的信息并输出
1 #include<stdio.h> 2 #include<string.h> 3 4 struct student{ 5 long num; 6 char name[20]; 7 int age; 8 char sex; 9 int score; 10 }; 11 12 int main(){ 13 struct student stu[3]; 14 int i; 15 printf("学生的数据是:长整型的学号,字符数组的名字,整型的年龄,字符型的性别,整型的分数。\n"); 16 for(i=0;i<3;i++){ 17 printf("Please input all data of student[%d]\n",i+1); 18 scanf("%ld,%s%d,%c,%d",&stu[i].num,&stu[i].name,&stu[i].age,&stu[i].sex,&stu[i].score); 19 } 20 printf("\n num\t name\t age\t sex\t score\n"); 21 for(i=0;i<3;i++){ 22 printf("%ld\t%s\t%d\t%4c\t%5d\n",stu[i].num,stu[i].name,stu[i].age,stu[i].sex,stu[i].score); 23 } 24 return 0; 25 }
本例用是scanf函数输入各成员时,除字符型数组外,其他变量都采用","分隔,而字符数组以回车作为输入的结束。
2)统计候选人得票数。假设有3名候选人,每次输入一个得票人的名字,要求最后输出每个人的得票总数。
1 #include<stdio.h> 2 #include<string.h> 3 4 struct person 5 { 6 char name[20]; 7 int count; 8 } leader[3]={"Hu",0,"Li",0,"Ma",0};//结构体数组的赋值 9 10 int main(){ 11 int i,j; 12 char leader_name[20]; 13 for(i=1;i<=5;i++){ 14 scanf("%s",leader_name); 15 for(j=0;j<3;j++){ 16 if(strcmp(leader_name ,leader[j].name )==0){ 17 leader[j].count++; 18 } 19 } 20 } 21 for(i=0;i<3;i++){ 22 printf("%5s:%d\n",leader[i].name,leader[i].count); 23 } 24 return 0; 25 }
(3)结构体指针
指向结构体的指针称为结构体指针变量。该变量存放结构体变量的起始地址。结构体指针变量也可以指向结构体数组中的元素。
1.结构体指针变量的定义
struct 结构体类型 *结构体指针
例如:
struct student stu1,*p=&stu1;
在访问结构体变量的成员时:p->num等价于(*p).num;
2.结构体数组指针
一个结构体指针变量可以指向结构体数组,即将结构体数组的起始地址赋给指针变量,这种指针就是结构体指针。
struct student stu1[10],*p=stu1;
(4)结构体类型数据在函数间的传递
函数间不仅可以传递简单变量、数组、指针这些类型的数据,还可以传递结构体类型的数据。函数间结构体类型数据的传递和普通变量一样,可以“按值传递”,也可以“按地址传递”。
1.结构体变量作为函数参数
用结构体变量作为函数实参传递数据:
1 #include<stdio.h> 2 3 struct Teacher{ 4 char name[20]; 5 float salary; 6 float reword; 7 float income; 8 }; 9 void display(struct Teacher p) 10 { 11 printf("\n%s %7.2f %7.2f %7.2f",p.name,p.salary,p.reword,p.income); 12 p.salary=3000; 13 p.reword=2000; 14 p.income=p.salary+p.reword; 15 printf("\n%s %7.2f %7.2f %7.2f",p.name,p.salary,p.reword,p.income); 16 } 17 int main(){ 18 struct Teacher teacher; 19 printf("\nInput name:"); 20 scanf("%s",teacher.name); 21 teacher.salary=2000; 22 teacher.reword=1000; 23 teacher.income=teacher.salary+teacher.reword; 24 display(teacher); 25 printf("\n%s %7.2f %7.2f %7.2f",teacher.name,teacher.salary,teacher.reword,teacher.income); 26 }
调用函数的实参与被调用函数的形参都是结构体变量名。
形参和实参的结构体类型相同,但运行时分配在不同的存储空间,因此,被调用函数不能修改调用函数的值。
2.结构体指针变量作为函数参数
1 #include<stdio.h> 2 3 struct student{ 4 long num; 5 char name[12]; 6 float score[3]; 7 }stb1={2006001,"LiYing",67.9,78.5,91.2},stb2={2006002,"YangLi",87.9,97.6,90.0}; 8 9 void output(struct student *p){ 10 printf("%ld\n%s\n%f\n%f\n%f\n",p->num,p->name,p->score[0],p->score[1],p->score[2]); 11 printf("\n"); 12 }; 13 14 int main(){ 15 output(&stb1); 16 output(&stb2); 17 return 0; 18 }
3.结构体数组作为函数参数
用结构体数组写一个联系人管理的c程序,包括增删改查,打印所有联系人信息五个功能!
1 #include<stdio.h> 2 #include<stdlib.h> 3 #include<string.h> 4 struct contacter{ 5 int num; 6 char name[15]; 7 unsigned long linknumber; 8 }; 9 struct contacter linkers[100]; 10 int count=0; 11 //程序开始时的提醒: 12 void FirstWarn(){ 13 printf("This is a c program used to manange your address book:\n"); 14 printf("Chose different choice,then you will get disfferent function:\n"); 15 printf("1.Establish and save a new contacter:\n"); 16 printf("2.Delete a contacter:\n"); 17 printf("3.Modify a contacter:\n"); 18 printf("4.Search information of a contacter:\n"); 19 printf("5.Print all of your contacters:\n"); 20 } 21 //第1个功能,用来新建一个联系人 22 void newContacters(){ 23 getchar(); 24 if(count==100){ 25 printf("There is no storage space any more!\n"); 26 }else if(count>=0&&count<100){ 27 printf("Please input name:\n"); 28 gets(linkers[count].name); 29 printf("please input his or her telephone number:\n"); 30 scanf("%ld",&linkers[count].linknumber); 31 linkers[count].num=14001+count; 32 count++; 33 } 34 } 35 //用来验证输入的命令数字 36 void validate(int* p){ 37 if(*p<1|*p>5){ 38 printf("Warning: No Such order!\n"); 39 printf("Please input a new order again:\n"); 40 scanf("%d",p); 41 validate(p); 42 } 43 } 44 //第2个功能,实现删除一个联系人 45 //(1)用号码来删除 46 void deleteByNumber(int num){ 47 int n; 48 n=num-14001; 49 for(int i=n;i<count;i++){ 50 strcpy(linkers[i].name,linkers[i+1].name); 51 linkers[i].linknumber=linkers[i+1].linknumber; 52 } 53 count--; 54 } 55 //(2)用名字来删除 56 void deleteByName(char name[]){ 57 for(int i=0;i<count;i++) 58 if(!strcmp(name,linkers[i].name)) 59 deleteByNumber(14001+i); 60 } 61 //第3个功能,实现输入名称修改联系人的信息 62 void modifyByName(char *name){ 63 for(int i=0;i<count;i++){ 64 if(!strcmp(name,linkers[i].name)){ 65 printf("Please input this linker's telephone:\n"); 66 scanf("%d",&linkers[i].linknumber); 67 } 68 printf("After modifing:num=%d name=%s telephone=%ld\n", 69 linkers[i].num,linkers[i].name,linkers[i].linknumber); 70 } 71 } 72 void modifyByNumber(int num){ 73 printf("Please input this linker's name\n"); 74 gets(linkers[num-14001].name); 75 printf("Please input this linkers' telephone\n"); 76 scanf("%ld",&linkers[num-14001].linknumber); 77 printf("After modifing:num=%d name=%s telephone=%ld\n", 78 linkers[num-14001].num,linkers[num-14001].name,linkers[num-14001].linknumber); 79 } 80 //第4个功能,实现查询功能 81 int searchByName(char *name,struct contacter *p,int count){ 82 for(int i=0;i<count;i++){ 83 if(!strcmp(name,p[i].name)){ 84 printf("The information of this person:\n"); 85 printf("num=%d,name=%15s,telephone=%ld\n",p[i].num,p[i].name,p[i].linknumber); 86 return 1; 87 } 88 } 89 return 0; 90 } 91 int searchByNumber(int num,struct contacter *p, int count){ 92 if(0<=num-14001&&num-14001<=count){ 93 printf("num=%d,name=%15s,telephone=%ld\n", 94 p[num-14001].num,p[num-14001].name,p[num-14001].linknumber); 95 } 96 } 97 //第5个功能,实现打印信息 98 void printAll(){ 99 for(int i=0;i<count;i++){ 100 printf("第%d位联系人:num=%d; name=%15s; telephone=%ld \n",i+1,linkers[i].num,linkers[i].name,linkers[i].linknumber); 101 } 102 } 103 int main(){ 104 //在启动程序时自动创建一个包含100个变量的结构体数组 105 //记录用户的选择 106 int choice; 107 FirstWarn(); 108 while(1){ 109 printf("Please input a order:\n"); 110 scanf("%d",&choice); 111 validate(&choice); 112 switch(choice){ 113 case 1:{ 114 newContacters(); 115 break; 116 } 117 case 2:{ 118 printf("Chose the way of deleting linkers:\n"); 119 printf("0.By name\n"); 120 printf("1.By Number:\n"); 121 int temp; 122 scanf("%d",&temp); 123 if(temp){ 124 int number; 125 printf("Input the number:\n"); 126 scanf("%d",&number); 127 deleteByNumber(number); 128 }else{ 129 char name[10]; 130 printf("Print the name of linker who you want to delete:\n"); 131 scanf("%s",name); 132 deleteByName(name); 133 } 134 break; 135 } 136 case 3:{ 137 int list=0; 138 printf("Chose a way to modyfy the contacter:\n"); 139 printf("1.Modify the telephone only(By name)!\n"); 140 printf("0.Modify both name and telephone(By number)!\n"); 141 scanf("%d",&list); 142 if(list){ 143 char name[10]; 144 printf("Please input the name:\n"); 145 getchar(); 146 gets(name); 147 modifyByName(name); 148 }else{ 149 int num; 150 printf("Please input the number:\n"); 151 scanf("%d",&num); 152 getchar(); 153 modifyByNumber(num); 154 } 155 break; 156 } 157 case 4: { 158 int list=0; 159 printf("Chose a way to search the contacter:\n"); 160 printf("1.search by name!\n"); 161 printf("0.search by number!\n"); 162 scanf("%d",&list); 163 if(list){ 164 char name[10]; 165 printf("Please input the name:\n"); 166 scanf("%s",name); 167 if(!searchByName(name,linkers,count)){ 168 printf("no such man!"); 169 } 170 }else{ 171 int num; 172 printf("Please input the person's number you want to search:\n"); 173 scanf("%d",&num); 174 searchByNumber(num,linkers,count); 175 } 176 break; 177 } 178 case 5: {printAll(); 179 break;} 180 default:printf("Errors!");break; 181 } 182 } 183 }
由于屏幕小,测试功能的截屏就是以上那么多!
总结遇到的问题:
1.在从键盘输入数据用scanf("%d",n),忘了加&,然而也不报错,检查半天,痛心!应为scanf("%d",&n);
2.本例用记录数组大小的数count,和结构体数组作为全局变量,事实上这样在每个函数中都可以自由更改它们的值。如果使用值传递的方式是不能更改count的值的,也不能更改结构体数组的值,操作起来非常不方便。
3.在用gets()函数,或者scanf()函数接收字符串时一定要确定前面是否有scanf()函数,如果有,确定用getchar();接收上面输入过程产生的回车符。
4.本例用long型数据存储电话号码,但貌似只能有8位整数,unsigned long貌似只能输入9位,无法达到11位的要求。这里就不在详细追究。事实上可以用字符数组来接收电话号码。