C语言_结构体总结

185 阅读6分钟

一起养成写作习惯!这是我参与「掘金日新计划 · 4 月更文挑战」的第24天,点击查看活动详情

当前文章介绍动态堆空间内存分配与释放,C语言结构体定义、初始化、赋值、结构体数组、结构体指针的相关知识点,最后通过一个学生管理系统综合练习结构体数组的使用。

1. 动态内存管理

C语言代码----->编译----->链接------>可执行的二进制文件(windows下xxx.exe) 二进制文件中的数据是如何摆放的? 文本数据段、静态数据段、全局数据段。

堆栈空间: 代码在运行的时候才有的空间。 栈空间: 系统负责申请,负责释放。比如: 函数形参变量、数组…… 堆空间: 程序员负责申请,负责释放。

 #include <stdlib.h>     //标准库头文件
 void *malloc(int size); //内存申请。 形参表示申请的空间大小,返回值:申请的空间的地址
 void free(void *p);     //内存释放。 形参就是要释放的空间首地址。

动态空间申请示例。

 动态空间申请
 #include "stdio.h"
 #include "string.h"
 #include <stdlib.h>
 int main()
 {
     int *p=malloc(sizeof(int)); //申请空间
     if(p!=NULL)
     {
         printf("申请的空间地址: 0x%X\n",p);
         *p=888;
         printf("%d\n",*p);
     }
     free(p); //释放空间
     return 0;
 }
 示例2:
 #include "stdio.h"
 #include "string.h"
 #include <stdlib.h>
 ​
 char *func(void)
 {
     char*str=malloc(100); //char str[100];
     if(str!=NULL)
     {
         strcpy(str,"1234567890");
         printf("子函数打印:%s\n",str);
         //free(str); //释放空间
         return str;
     }
     else
     {
         return NULL;
     }
 }
 ​
 int main()
 {
     char *p=func();
     printf("主函数打印:%s\n",p);
     return 0;
 }

2. 结构体

2.1 定义语法

结构体的概念: 可存放不同数据类型的集合。 比如: 存放一个班级学生的信息。 可以使用一个结构体存放一个学生的信息。 一个结构体数组存放整个班级的学习信息。 数组的概念: 可存放相同数据类型的集合。

结构体的定义语法:

 //声明一种新类型-----数据类型
 struct <结构体的名称>
 {
     <结构体的成员>1;
 <结构体的成员>2;
 …………
 };  //最后有分号结束
 ​
 struct MyStruct 
 {
     char a;
     int b;
     float c;
     char str[100];
 };

2.2 定义示例

结构体如何赋值? 如何访问结构体内部成员

 #include "stdio.h"
 #include "string.h"
 #include <stdlib.h>
 ​
 //定义结构体数据类型
 struct MyStruct
 {
     char a;
     int b;
     float c;
     char str[100];
 };
 int main()
 {
     struct MyStruct data={'A',123,456.789,"abcd"}; //data就是结构体类型的变量
     //结构体变量访问内部成员的语法:  . 点运算符
     printf("%c\n",data.a);
     printf("%d\n",data.b);
     printf("%f\n",data.c);
     printf("%s\n",data.str);
     return 0;
 }

2.3 初始化

 #include "stdio.h"
 #include "string.h"
 #include <stdlib.h>
 ​
 //定义结构体数据类型
 struct MyStruct
 {
     char a;
     int b;
     float c;
     char str[100];
 }data={'A',123,456.789,"abcd"}; //data就是结构体类型的变量
 ​
 int main()
 {
     //结构体变量访问内部成员的语法:  . 点运算符
     printf("%c\n",data.a);
     printf("%d\n",data.b);
     printf("%f\n",data.c);
     printf("%s\n",data.str);
     return 0;
 }

2.4 结构体赋值

   //结构体变量访问内部成员的语法:  . 点运算符
 #include "stdio.h"
 #include "string.h"
 #include <stdlib.h>
 ​
 //定义结构体数据类型
 struct MyStruct
 {
     char a;
     int b;
     float c;
     char str[100];
 }; 
 ​
 int main()
 {
     struct MyStruct data;//data就是结构体类型的变量
     //成员单独赋值
     data.a='A';
     data.b=123;
     data.c=456.789;
     strcpy(data.str,"abcd"); //数组赋值
 ​
     //结构体变量访问内部成员的语法:  . 点运算符
     printf("%c\n",data.a);
     printf("%d\n",data.b);
     printf("%f\n",data.c);
     printf("%s\n",data.str);
     return 0;
 }
 ​

2.5 结构体数组

 结构体赋值分为两种标准:  C89 、C99
 结构体数组
 #include "stdio.h"
 #include "string.h"
 #include <stdlib.h>
 ​
 //定义结构体数据类型
 struct MyStruct
 {
     char a;
     int b;
     float c;
     char str[100];
 }; 
 ​
 int main()
 {
     struct MyStruct data[100];//data就是结构体数组类型变量
     struct MyStruct data2[50];
 ​
     //成员单独赋值
     data[0].a='A';
     data[0].b=123;
     data[0].c=456.789;
     strcpy(data[0].str,"abcd"); //数组赋值
 ​
     //结构体变量访问内部成员的语法:  . 点运算符
     printf("%c\n",data[0].a);
     printf("%d\n",data[0].b);
     printf("%f\n",data[0].c);
     printf("%s\n",data[0].str);
     return 0;
 }

2.6 结构体指针赋值

 #include "stdio.h"
 #include "string.h"
 #include <stdlib.h>
 //定义结构体数据类型
 struct MyStruct
 {
     char a;
     int b;
     float c;
     char str[100];
 }; 
 ​
 int main()
 {
     //struct MyStruct buff[100];
     //struct MyStruct *data=buff; //结构体指针类型变量
 ​
     struct MyStruct *data=malloc(sizeof(struct MyStruct));
     data->a='A';
     data->b=123;
     data->c=456.789;
     strcpy(data->str,"abcd");
 ​
     //结构体指针访问内部成员的变量 通过  ->  运算符。 
     printf("%c\n",data->a);
     printf("%d\n",data->b);
     printf("%f\n",data->c);
     printf("%s\n",data->str);
     return 0;
 }

3. 学生管理系统

作业: 学生管理系统

需求: (每一个功能都是使用函数进行封装) 1.实现从键盘上录入学生信息。 (姓名、性别、学号、成绩、电话号码) 2.将结构体里的学生信息全部打印出来。 3.实现根据学生的姓名或者学号查找学生,查找到之后打印出学生的具体信息。 4.根据学生的成绩对学生信息进行排序。 5.根据学号删除学生信息。

示例:

 ​
 #include "stdio.h"
 #include "string.h"
 #include <stdlib.h>
 ​
 //定义存放学生信息的结构体类型
 struct StuDentInfo
 {
     char Name[20]; //姓名
     int number;    //学号
     char phone[20];//电话号码
 }; 
 ​
 //全局变量区域
 unsigned int  StuDentCnt=0; //记录已经录入的全部学生数量
 ​
 //函数声明区域
 void PrintStuDentInfoList(void);
 void InputStuDentInfo(struct StuDentInfo*info);
 void FindStuDentInfo(struct StuDentInfo*info);
 void SortStuDentInfo(struct StuDentInfo*info);
 void PrintStuDentInfo(struct StuDentInfo*info);
 ​
 int main()
 {
     struct StuDentInfo data[100]; //可以100位学生的信息
     int number;
     while(1)
     {
         PrintStuDentInfoList(); //打印功能列表
         scanf("%d",&number);
         printf("\n");
         switch(number)
         {
         case 1:
             InputStuDentInfo(data);
             break;
         case 2:
             FindStuDentInfo(data);
             break;
         case 3:
             SortStuDentInfo(data);
             break;
         case 4:
             PrintStuDentInfo(data);
             break;
         case 5:
             break;
         default:
             printf("选择错误!\n\n");
             break;
         }
     }
     return 0;
 }
 ​
 /*
 函数功能: 打印学生管理系统的功能列表
 */
 void PrintStuDentInfoList(void)
 {
     printf("\n--------------学生管理系统功能列表----------------\n");
     printf("1. 录入学生信息\n");
     printf("2. 根据学号查找学生信息\n");
     printf("3. 根据学号排序\n");
     printf("4. 打印所有学生信息\n");
     printf("5. 删除指定的学生信息\n");
     printf("请选择功能序号:");
 }
 ​
 /*
 函数功能: 录入学生信息
 */
 void InputStuDentInfo(struct StuDentInfo*info)
 {
     printf("输入学生姓名:");
     scanf("%s",info[StuDentCnt].Name);
     printf("输入学号:");
     scanf("%d",&info[StuDentCnt].number);
     printf("输入电话号码:");
     scanf("%s",info[StuDentCnt].phone);
     StuDentCnt++; //数量自增
 }
 ​
 /*
 函数功能: 查找学生信息
 */
 void FindStuDentInfo(struct StuDentInfo*info)
 {
     int num,i;
     printf("输入查找的学号:");
     scanf("%d",&num);
     for(i=0; i<StuDentCnt; i++)
     {
         if(info[i].number==num)
         {
             printf("信息查找成功,该学生的信息如下:\n");
             printf("姓名:%s\n",info[i].Name);
             printf("学号:%d\n",info[i].number);
             printf("电话号码:%s\n",info[i].phone);
             printf("\n");
             break;
         }
     }
     if(i==StuDentCnt)
     {
         printf("----------%d学号不存在!---------\n",num);
     }
 }
 ​
 /*
 函数功能: 根据学号排序
 */
 void SortStuDentInfo(struct StuDentInfo*info)
 {
     int i,j;
     struct StuDentInfo tmp; //保存临时信息
     for(i=0; i<StuDentCnt-1; i++)
     {
         for(j=0;j<StuDentCnt-i-1;j++)
         {
             if(info[j].number>info[j+1].number)
             {
                 tmp=info[j];
                 info[j]=info[j+1];
                 info[j+1]=tmp;
             }
         }
     }
 }
 ​
 /*
 函数功能: 打印所有学生信息
 */
 void PrintStuDentInfo(struct StuDentInfo*info)
 {
     int i=0;
     printf("-----------所有学生的信息列表------------\n");
     for(i=0;i<StuDentCnt;i++)
     {
         printf("姓名:%s\n",info[i].Name);
         printf("学号:%d\n",info[i].number);
         printf("电话号码:%s\n",info[i].phone);
         printf("\n");
     }
 }