收藏 分销(赏)

C语言教学课件:c语言结构体.ppt

上传人:可**** 文档编号:7906692 上传时间:2025-01-26 格式:PPT 页数:100 大小:1.48MB
下载 相关 举报
C语言教学课件:c语言结构体.ppt_第1页
第1页 / 共100页
C语言教学课件:c语言结构体.ppt_第2页
第2页 / 共100页
C语言教学课件:c语言结构体.ppt_第3页
第3页 / 共100页
C语言教学课件:c语言结构体.ppt_第4页
第4页 / 共100页
C语言教学课件:c语言结构体.ppt_第5页
第5页 / 共100页
点击查看更多>>
资源描述

1、,单击此处编辑母版文本样式,第二级,第三级,第四级,第五级,*,单击此处编辑母版标题样式,结构体,要点,结构体的概念,结构体的定义和引用,结构体数组,主要内容,11.1,概述,11.2,定义结构体类型变量的方法,11.3,结构体变量的引用,11.4,结构体变量的初始化,11.5,结构体数组,11.,指向结构体类型数据的指针,11.7,用指针处理链表,11.8,共用体,11.9,枚举类型,11.10,用,typedef,定义类型,11.1,概述,问题定义:,有时需要将不同类型的数据组合成一个有机,的整体,以便于引用。,如:,一个学生有学号,/,姓名,/,性别,/,年龄,/,地址等属性,int n

2、um;char name20;char sex;,int age;int char addr30;,应当把它们组织成一个组合项,在一个组合,项中包含若干个类型不同(当然也可以相同),的数据项。,图,11-1,100101 Li Fun M 18 87.5 Beijing,Num name sex age score addr,11.1,概述,声明一个结构体类型的一般形式为:,struct,结构体名,成员表列;,如:,struct,student,int,num;,char,name20;,char,sex;,int,age;,float,score;,char,addr30;,结构体名,类型名

3、,成员名,11.2,定义结构体类型变量的方法,可以采取以下,3,种方法定义结构体类型变量:,(1),先声明结构体类型再定义变量名,例如:,struct student,student1,student2,;,|,结构体类型名 结构体变量名,定义了,student1,和,student2,为,struct student,类型的变量,即它们具有,struct student,类型的结构,.,图,11-2,student1,100101 ZhangXin M 19 90.5 Shanghai,100102 WangLi F 20 98 Beijing,student2,11.2,定义结构体类型变量

4、的方法,在定义了结构体变量后,系统会为之分配内存单元。,例如,:,student1,和,student2,在内存中各占,59,个字节(,2+20+1+2+4+30=59,)。,注意:,将一个变量定义为标准类型(基本数据类型)与定义为结构体类型不同之处在于后者不仅要求指定变量为结构体类型,而且要求指定为某一特定的结构体类型,因为可以定义出许许多多种具体的结构体类型。,11.2,定义结构体类型变量的方法,(2),在声明类型的同时定义变量,这种形式的定义的一般形式为,:,struct,结构体名,成员表列,变量名表列;,11.2,定义结构体类型变量的方法,例如:,struct,student,int,

5、num,;,char,name20,;,char,sex,;,int,age,;,float,score,;,char,addr30,;,student1,student2;,它的作用与第一种方法相同,即定义了两个,struct student,类型的变量,student1,student2,11.2,定义结构体类型变量的方法,(3),直接定义结构体类型变量,其一般形式为,:,struct,成员表列,变量名表列;,即不出现结构体名。,注意:,(1),类型与变量是不同的概念,不要混同。只能对变量赋值、存取或运算,而不能对一个类型赋值、存取或运算。在编译时,对类型是不分配空间的,只对变量分配空间。

6、,注意:,(2),对结构体中的成员(即,“,域,”,),可以单独使用,它的作用与地位相当于普通变量。,(3),成员也可以是一个结构体变量。,(4),成员名可以与程序中的变量名相同,二者不代表同一对象。,11.2,定义结构体类型变量的方法,例如:,struct,student,/*,声明一个结构体类型*,/,int,num,;,char,name20,;,char,sex,;,int,age,;,float,score,;,struct,date,birthday,;,/*birthday,是,struct date,类型*,/,char,addr30,;,student1,student2;,

7、先声明一个,struct date,类型,它代表“日期”,包括,3,个成员:,month,(月)、,day,(日)、,year,(年)。然后在声明,struct student,类型时,将成员,birthday,指定为,struct date,类型。,图,11-3,birthday addr,Num name sex age,Month day year,11.3,结构体变量的引用,在定义了结构体变量以后,当然可以引用这个变量。但应遵守以下规则,:,(1),不能将一个结构体变量作为一个整体进行输入和输出。,例如,:,已定义,student1,和,student2,为结构体变量并且它们已有值。,

8、printf(%d,%s,%c,%d,%f,%,n,student1);,11.3,结构体变量的引用,引用结构体变量中成员的方式为,结构体变量名,.,成员名,例如,,student1.num,表示,student1,变量中的,num,成员,即,student1,的,num(,学号,),项。可以对变量的成员赋值,例如,:student1.num=10010;,“,.,”,是成员,(,分量,),运算符,它在所有的运算符中优先级最高,因此可以把,student1.num,作为一个整体来看待。上面赋值语句的作用是将整数,10010,赋给,student1,变量中的成员,num,。,11.3,结构体变量

9、的引用,(2),如果成员本身又属一个结构体类型,则要用若干个成员运算符,一级一级地找到最低的一级的成员。只能对最低级的成员进行赋值或存取以及运算。,例如,:,对上面定义的结构体变量,student1,可以这样访问各成员,:,student1.num,student1.birthday.month,注意:,不能用,student1.birthday,来访问,student1,变量中的成员,birthday,因为,birthday,本身是一个结构体变量。,11.3,结构体变量的引用,(3),对结构体变量的成员可以像普通变量一样进行各种运算(根据其类型决定可以进行的运算)。,例如:,student2

10、.score=student1.score;,sum=student1.score+student2.score;,student1.age+;,+student2.age;,由于“”运算符的优先级最高,因此是对进行自加运算,而不是先对进行自加运算。,11.3,结构体变量的引用,(4),可以引用结构体变量成员的地址,也可以引用结构体变量的地址。,例如:,scanf(%d,,,(输入,student1.num,的值),printf(%o,,,student1,);,(输出,student1,的首地址),11.3,结构体变量的引用,但不能用以下语句整体读入结构体变量,,例如:,scanf,(,%d

11、,,,s,,,c,,,d,,,f,,,s,,,student1,);,结构体变量的地址主要用作函数参数,,传递结构体变量的地址。,11.,结构体变量的初始化,但不能用以下语句整体读入结构体变量,,例如:,scanf,(,%d,,,s,,,c,,,d,,,f,,,s,,,student1,);,结构体变量的地址主要用作函数参数,传递结构体变量的地址。,例,11.1,对结构体变量初始化,.,#include void main,(),struct student,long int num,;,char name20;char sex,;,char addr20,;,a=10101,,,LiLin,

12、,,M,,,123 Beijing Road,;,/*,对结构体变量,a,赋初值*,/,printf(No.:%ld,nname:%s,nsex:%c,naddress:%s,n,,,a.num,,,a.name,,,a.sex,,,a.addr);,运行结果:,No.,:,10101,name,:,LiLin,sex,:,address,:,123 Beijing Road,11.5,结构体数组,一个结构体变量中可以存放一组数,据(如一个学生的学号、姓名、成绩等,数据)。如果有个学生的数据需要,参加运算,显然应该用数组,这就是结,构体数组。结构体数组与以前介绍过的,数值型数组不同之处在于每个

13、数组元素,都是一个结构体类型的数据,它们都分,别包括各个成员(分量)项。,11.5,结构体数组,11.5.1,定义结构体数组,和定义结构体变量的方法相仿,只需说明其为数组即可。,例如:,struct student,int num;char name20;char sex;int age;,float score;char addr30;,;,struct student3;,以上定义了一个数组,stu,,数组有个元素,均为,struct student,类型数据。,11.5,结构体数组,也可以直接定义一个结构体数组,,例如,:,struct student,int num;,;stu3;,或

14、:,strcut student,int num;,;stu3;,图,11-4,11.5,结构体数组,11.5.2,结构体数组的初始化,与其他类型的数组一样,对结构体数组可以初始化。,例如,:,struct student,int num;char name20,;,char sex,;,int age,;,float score;char addr30,;,stu,2,10101,,,LiLin,,,M,,,18,,,87.5,,,103 BeijingRoad,,,10102,,,Zhang Fun,,,M,,,19,,,99,,,130 Shanghai Road,;,图,11-5,11

15、.5,结构体数组,当然,数组的初始化也可以用以下形式:,struct student,int num,;,;,struct student,str,;,即先声明结构体类型,然后定义数组为该结构体类型,在定义数组时初始化。,结构体数组初始化的一般形式是在定义数组的后面加上“初值表列;”。,11.5,结构体数组,11.5.3,结构体数组应用举例,例,11.2,对候选人得票的统计程序。设有,3,个候选人,每次输入一个得票的候选人的名字,要求最后输出各人得票结果。,#include#include struct person char name20;int count;leader3=,“,Li,”,

16、0,“,Zhang,”,0,“,Fun,”,0;,例,11.2,void main()int i,j;char leader_name20;for(i=1;i=10;i+)scanf(,“,%s,”,leader_name);for(j=0;j3;j+)if(strcmp(leader_name,leaderj.name)=0)leaderj.count+;printf(,“,n,”,);for(i=0;i,成员名,其中,-,称为指向运算符。,请分析以下几种运算:,-,得到指向的结构体变量中的成员的值。,-,得到指向的结构体变量中的成员的值,用完该值后使它加。,-,得到指向的结构体变量中的成员

17、的值加,然后再使用它。,11.6,指向结构体类型数据的指针,11.6.2,指向结构体数组的指针,例,11.4,指向结构体数组的指针的应用,#include struct studentint num;char name20;char sex;int age;struct student stu3=,10101,,,Li Lin,,,M,,,18,,,10102,,,Zhang Fun,,,M,,,19,,,10104,,,WangMing,,,F,,,20,;void main()struct student*p;printf(No.Name sex age,);,for,(,str,;,st

18、r,;,p,),printf(%5d%-20s%2c%4d,n,,,p-num,,,p-name,,,p-sex,,,p-age);,运行结果:,LiLin,18,Zhang Fun,19,WangMing,20,11.6,指向结构体类型数据的指针,程序分析:,是指向,struct student,结构体类型数据的指针变量。在,for,语句中先使的初值为,stu,,也就是数组,stu,第一个元素的起始地址。在第一次循环中输出,stu0,的各个成员值。然后执行,使自加。加意味着,p,所增加的值为结构体数组,stu,的一个元素所占的字节数。执行,+,后,p,的值等于,stu,1,,指向,stu1,

19、。在第二次循环中输出,stu1,的各成员值。在执行后,p,的值等于,stu+2,,再输出,stu 2,的各成员值。在执行,+,后,的值变为,stu+,,已不再小于,stu+3,了,不再执行循环。,图,11-8,11.6,指向结构体类型数据的指针,注意:,(1),如果的初值为,stu,,即指向第一个元素,则加后,p,就指向下一个元素。,例如,:,(+p),-num,先使自加,然后得到它指向的元素中的,num,成员值(即,10102,)。,(p+)-num,先得到,-num,的值(即,10101,),然后使自加,指向,stu1,。,请注意以上二者的不同。,11.6,指向结构体类型数据的指针,注意:

20、,(2),程序已定义了是一个指向,struct student,类型数据的指针变量,它用来指向一个,struct student,类型的数据,不应用来指向,stu,数组元素中的某一成员。,例如,:,1,a,;,如果要将某一成员的地址赋给,p,,可以用强制类型转换,先将成员的地址转换成,p,的类型。,例如:(*),0,a,;,11.6,指向结构体类型数据的指针,11.6.3,用结构体变量和指向结构体的指针,作函数参数,将一个结构体变量的值传递给另一个函数,有,3,个方法,:,用结构体变量的成员作参数。,(2),用结构体变量作实参。,(3),用指向结构体变量(或数组)的指针作实参,将结构体变量(或

21、数组)的地址传给形参。,11.6,指向结构体类型数据的指针,11.6.2,指向结构体数组的指针,例,11.5,有一个结构体变量,stu,,内含学生学号、姓名和,3,门课程的成绩。要求在,main,函数中赋予值,在另一函数,print,中将它们输出。今用结构体变量作函数参数。,#include struct student int num;char name20;float score3;,11.6,指向结构体类型数据的指针,void main()void print(struct student);struct student stu;stu.num=12345;strcpy(stu.name

22、,LiLin;stu.score0=67.5;stu.score1=89;stu.score2 =78.6);print(stu);void print(struct student stu)printf(FORMAT,,,stu.num,,,stu.name,,,stu.score0,,,stu.score1,,,stu.score2,);,printf,(,n,);,运行结果:,67.500000,89.000000,78.599998,例,11.6,将上题改用指向结构体变量的指针作实参。,#include struct student int num;char name20;float

23、score3;stu=12345,LiLi,67.5,89,78.6;void main()void print(struct student*);,/*,形参类型修改成指向结构体的指针变量*,/,print(,/*,实参改为,stu,的起始地址*,/,void print(struct student*p),/*,形参类型修改了*,/,printf(FORMAT,,,p-num,,,p-name,,,p-score,0,,,p-score,1,,,p-score,2,);,/*,用指针变量调用各成员的值*,/,printf,(,);,运行结果:,67.500000,89.000000,78.

24、599998,11.6,指向结构体类型数据的指针,程序分析:,此程序改用在定义结构体变量,stu,时赋初值,这样程序可简化些。,print,函数中的形参被定义为指向,struct student,类型数据的指针变量。注意在调用,print,函数时,用结构体变量,str,的起始地址,stu,作实参。在调用函数时将该地址传送给形参,p(p,是指针变量)。这样就指向,stu,。在,print,函数中输出所指向的结构体变量的各个成员值,它们也就是,stu,的成员值。,main,函数中的对各成员赋值也可以改用,scanf,函数输入。,图,11-9,11.7,用指针处理链表,11.7.1,链表概述,链表是

25、一种常见的重要的数据结构,是动,态地进行存储分配的一种结构。,链表的组成:,头指针:存放一个地址,该地址指向一个元素,结点:用户需要的实际数据和链接节点的指针,图,11-10,11.7,用指针处理链表,用结构体建立链表:,struct,student,int,num,;,float,score,;,struct,student*next,;,;,其中成员,num,和,score,用来存放结点中的有用数据(用户需要用到的数据),,next,是指针类型的成员,它指向,struct student,类型数据(这就是,next,所在的结构体类型),图,11-11,11.7,用指针处理链表,11.7.2

26、,简单链表,#include#define NULL 0,struct student long num;float score;struct student*next;,main()struct student a,b,c,*head,*p;a.num=99101;a.score=89.5;b.num=99103;b.score=90;c.num=99107;c.score=85;head=,运行结果:,1010189.5,1010390.0,1010785.0,11.7,用指针处理链表,程序分析:,开始时使,head,指向,a,结点,,a.next,指向,b,结点,,b.next,指向,c

27、,结点,这就构成链表关系。,“,c.next=NULL,”,的作用是使,c.next,不指向任何有用的存储单元。在输出链表时要借助,p,,先使,p,指向,a,结点,然后输出,a,结点中的数据,,“,p=p-next,”,是为输出下一个结点作准备。,p-next,的值是,b,结点的地址,因此执行,“,p=p-next,”,后,p,就指向,b,结点,所以在下一次循环时输出的是,b,结点中的数据。,11.7,用指针处理链表,11.7.3,处理动态链表所需的函数,库函数提供动态地开辟和释放存储单元的,有关函数:,malloc,函数,其函数原型为,void*malloc(unsigned int siz

28、e);,其,作用是在内存的动态存储区中分配一个长度为,size,的连续空间。此函数的值(即“返回值”),是一个指向分配域起始地址的指针(类型为,void,)。如果此函数未能成功地执行(例如内,存空间不足),则返回空指针,(NULL),。,11.7,用指针处理链表,(2)calloc,函数,其函数原型为,void*calloc,(,unsigned,,,unsigned size,),;,其作用是在内存的动态存储区,中分配个长度为,size,的连续空间。函数返回,一个指向分配域起始地址的指针;如果分配不,成功,返回,NULL,。,用,calloc,函数可以为一维数组开辟动态存,储空间,,n,为数

29、组元素个数,每个元素长度为,Size,。,11.7,用指针处理链表,(3)free,函数,其函数原型为,void free,(,void*p,),;,其作用,是释放由指向的内存区,使这部分内存区能,被其他变量使用。是最近一次调用,calloc,或,malloc,函数时返回的值。,free,函数无返回值。,以前的版本提供的,malloc,和,calloc,函数,得到的是指向字符型数据的指针。,ANSI,提,供的,malloc,和,calloc,函数规定为,void,类型。,11.7,用指针处理链表,11.7.4,建立动态链表,所谓建立动态链表是指在程序执行过程中从,无到有地建立起一个链表,即一个

30、一个地开辟结,点和输入各结点数据,并建立起前后相链的关系,例,11.5,写一函数建立一个有,3,名学生数据的单向动,态链表。,算法如图,图,11-12,11.7,用指针处理链表,算法的实现:,我们约定学号不会为零,如果输入的学号为,,则表示建立链表的过程完成,该结点不应连,接到链表中。,如果输入的,p1-num,不等于,则输入的是第,一个结点数据(,n=1,),令,head,p1,,即把,p1,的值,赋给,head,,也就是使,head,也指向新开辟的结点,p1,所指向的新开辟的结点就成为链表中第一个结点,图,11-13,11.7,用指针处理链表,算法的实现:,再开辟另一个结点并使,p1,指向

31、它,接着输入该,结点的数据,.,如果输入的,p1-num,,则应链入第个结点,(,n=2),将新结点的地址赋给第一个结点的,next,成员,.,接着使,也就是使指向刚才建,立的结点,图,11-14,11.7,用指针处理链表,算法的实现:,再开辟一个结点并使,p1,指向它,并输入该结点的,数据。,在第三次循环中,由于(),又,将的值赋给,-,,也就是将第,个结点连接到第个结点之后,并使,,使指向最后一个结点,.,图,11-15,11.7,用指针处理链表,算法的实现:,再开辟一个新结点,并使,p1,指向它,输入该结,点的数据。由于,p1-num,的值为,不再执行循环,,此新结点不应被连接到链表中,

32、.,将,NULL,赋给,p2-next.,建立链表过程至此结束,,p1,最后所指的结点,未链入链表中,第三个结点的,next,成员的值,为,NULL,,它不指向任何结点。,图,11-16,11.7,用指针处理链表,建立链表的函数如下,:,#include,#include,#define NULL 0,/,令,NULL,代表,用它表示,“,空地址,#define LEN sizeof(struct student),/,令,LEN,代表,struct/student,类型数据的长度,struct student,long num;,float score;struct student*next

33、;,;int n;,/n,为全局变量,本文件模块中各函数均可使用它,11.7,用指针处理链表,struct student*creat(),struct student*head;struct student*p1,*p2;n=0;,p1=p2=(struct student*)malloc(LEN);,scanf(%ld,%f,head=NULL;,while(p1-num!=0),n=n+1;if(n=1)head=p1;else p2-next=p1;,p2=p1;p1=(struct student*)malloc(LEN);,scanf(%ld,%f,p2-next=NULL;ret

34、urn(head);,11.7,用指针处理链表,11.7.5,输出链表,首先要知道链表第一个结点的地址,也就是,要知道,head,的值。然后设一个指针变量,p,先指向,第一个结点,输出所指的结点,然后使后移,一个结点,再输出,直到链表的尾结点。,图,11-17,11-18,11.7,用指针处理链表,例,1,9,编写一个输出链表的函数,print.,void print(struct student*head),struct student*p;,printf(nNow,These%d records are:n,n);p=head;,if(head!=NULL),do,printf(%ld%5

35、.1fn,p-num,p-score);,p=p-next;,while(p!=NULL);,11.7,用指针处理链表,11.7.6,对链表的删除操作,从一个动态链表中删去一个结点,并不是真,正从内存中把它抹掉,而是把它从链表中分离开,来,只要撤销原来的链接关系即可。,图,11-19,11.7,用指针处理链表,例,11.10,写一函数以删除动态链表中指定的结点,.,解题思路,:,从,p,指向的第一个结点开始,检查该结点中的,num,值是否等于输入的要求删除的那个学号。如果,相等就将该结点删除,如不相等,就将,p,后移一个,结点,再如此进行下去,直到遇到表尾为止。,11.7,用指针处理链表,可以

36、设两个指针变量,p1,和,p2,,先使,p1,指向,第一个结点。,如果要删除的不是第一个结点,则使,p1,后,移指向下一个结点,(,将,p1-next,赋给,p1),,在此,之前应将,p1,的值赋给,p2,,使,p2,指向刚才检查,过的那个结点。,11.7,用指针处理链表,注意:,要删的是第一个结点(的值等于的值,如图,1-,0,()那样),则应将,-,赋给。这时指向原来的第二个结点。第一个结点虽然仍存在,但它已与链表脱离,因为链表中没有一个结点或头指针指向它。虽然还指向它,它仍指向第二个结点,但仍无济于事,现在链表的第一个结点是原来的第二个结点,原来第一个结点已,“,丢失,”,,即不再是链表

37、中的一部分了。,11.7,用指针处理链表,注意:,如果要删除的不是第一个结点,则将,-,赋给,-,,见图,1,0,()。,-,原来指向指向的结点(图中第二个结点),现在,-,改为指向,-,所指向的结点(图中第三个结点)。所指向的结点不再是链表的一部分。,还需要考虑链表是空表(无结点)和链表中找不到要删除的结点的情况。,11.7,用指针处理链表,图,11-20,11.7,用指针处理链表,算法:,图,11-21,11.7,用指针处理链表,删除结点的函数,del:,struct student*del(struct student*head,long num),struct student*p1,*

38、p2;,if(head=NULL)printf(nlist null!n);goto end;,p1=head;,while(num!=p1-num,if(num=p1-num),if(p1=head)head=p1-next;,else p2-next=p1-next;,printf(delete:%ldn,num);n=n-1;,else printf(%ld not been found!n,num);,end;return(head);,11.7,用指针处理链表,11.7.7,对链表的插入操作,对链表的插入是指将一个结点插入到一个已有的链表中。,为了能做到正确插入,必须解决两个问题:,

39、怎样找到插入的位置;,怎样实现插入。,11.7,用指针处理链表,先用指针变量,p0,指向待插入的结点,,p1,指向第,一个结点。,将,p0-num,与,p1-num,相比较,如果,p0-num,p1-num,,则待插入的结点不应插在,p1,所指的,结点之前。此时将,p1,后移,并使,p2,指向刚才,p1,所指的结点。,11.7,用指针处理链表,再将,p1-num,与,p0-num,比,如果仍然是,p0-num,大,则应使,p1,继续后移,直到,p0-p1-num,为止。,这时将,p0,所指的结点插到,p1,所指结点之前。但是如,果,p1,所指的已是表尾结点,则,p1,就不应后移了。如,果,p0

40、-num,比所有结点的,num,都大,则应将,p0,所指,的结点插到链表末尾。,如果插入的位置既不在第一个结点之前,又不,在表尾结点之后,则将,p0,的值赋给,p2-next,,使,p2-next,指向待插入的结点,然后将,p1,的值赋给,p0-next,,使得,p0-next,指向,p1,指向的变量。,11.7,用指针处理链表,如果插入位置为第一个结点之前,(,即,p1,等于,head,时,),,则将,p0,赋给,head,,将,p1,赋给,p0-next,如果要插到表尾之后,应将,p0,赋给,p1-next,,,NULL,赋给,p0-next,图,11-22,11.7,用指针处理链表,算法

41、:,图,11-23,11.7,用指针处理链表,例,11.11,插入结点的函数,insert,如下。,struct student*insert(struct student*head,struct student*stud),struct student*p0,*p1,*p2;,p1=head;p0=stud;if(head=NULL),head=p0;p0-next=NULL;,elsewhile(p0-nump1-num)&(p1-next!=NULL),p2=p1;p1=p1-next;,if(p0-numnum)if(head=p1)head=p0;,else p2-next=p0;p

42、0-next=p1;,else p1-next=p0;p0-next=NULL;,n=n+1;return(head);,11.7,用指针处理链表,11.7.8,对链表的综合操作,将以上建立、输出、删除、插入的函数组织,在一个,C,程序中,用函数作主调函数。,void main()struct student*head,stu;long del_num;prinf(intput records:n);head=creat();print(head);printf(n intput the deleted number:n);scanf(%ld,11.7,用指针处理链表,此程序运行结果是正确的。

43、它只删除一个结,点,插入一个结点。但如果想再插入一个结点,,重复写上程序最后,4,行,共插入两个结点,运行结,果却是错误的。,Input records,:(建立链表),10,,,10,,,10,,,,,11.7,用指针处理链表,Now,these 3 records are,:,10,10,10,intput the deleted number,:,10103,(删除),:,10,Now,these 4 records are,:,10,10,11.7,用指针处理链表,input the inserted record,(插入第一个结点),10102,,,90,Now,these 3 re

44、cords are,:,10,10,10,input the inserted record,(插入第二个结点),10104,,,99,Now,these 4 records are,:,10,10,10,10,11.7,用指针处理链表,出现以上结果的原因是:,stu,是一个有固定地址的结构体变量。第一次把,stu,结点插入到链表中,第二次若再用它来插入第二个结点,就把第一次结点的数据冲掉了,实际上并没有开辟两个结点。为了解决这个问题,必须在每插入一个结点时新开辟一个内存区。我们修改,main,函数,使之能删除多个结点(直到输入要删的学号为,0,),能插入多个结点(直到输入要插入的学号为,0,

45、)。,11.7,用指针处理链表,main()struct student*head,*stu;long del_num;printf(input records:n);head=creat();print(head);printf(ninput the deleted number:);scanf(%ld,11.7,用指针处理链表,stu,定义为指针变量,在需要插入时先用,malloc,函数开辟一个内存区,将其起始地址经强,制类型转换后赋给,stu,,然后输入此结构体变量中,各成员的值。对不同的插入对象,,stu,的值是不同,的,每次指向一个新的,struct student,变量。在,调用,

46、insert,函数时,实参为,head,和,stu,,将已建立,的链表起始地址传给,insert,函数的形参,将,stu,(,即新开辟的单元的地址)传给形参,stud,,返回的,函数值是经过插入之后的链表的头指针(地址),11.7,用指针处理链表,运行结果:,:,10,,,10,,,10,,,,,,:,10,10,10,11.7,用指针处理链表,intput the deleted number,10103,(删除),:,10,Now,these 4 records are,10,9,10,intput the deleted number,10103,(删除),:,10,5,Now,thes

47、e 4 records are,10,9,11.7,用指针处理链表,intput the deleted number:0,input the inserted record,10104,,,87,Now,these 3 records are,10101 99.0,10104 87,input the inserted record,10106,,,65,Now,these 3 records are,10101 99.0,10104 87,10106 65.0,11.8,共用体,11.8.1,共用体的概念,使几个不同的变量共占同一段内存的结构,称为,“,共用体,”,类型的结构。,定义共用体

48、类型变量的一般形式为:,union,共用体名,成员表列,变量表列;,图,11-24,11.8,共用体,例如:,union data union data,int i,;,int i;,char ch,;,或,char ch;,float f,;,float f,;,a,b,c;union data a,b,c;,11.8,共用体,共用体和结构体的比较:,结构体变量所占内存长度是各成员占的内存长度之和。每个成员分别占有其自己的内存单元。,共用体变量所占的内存长度等于最长的成员的长度。,共用体和结构体的比较:,结构体变量所占内存长度是各成员占的内存长度之和。每个成员分别占有其自己的内存单元。,共用

49、体变量所占的内存长度等于最长的成员的长度。,例如,:,上面定义的,“,共用体,”,变量、各占,个字节(因为一个实型变量占个字节),而不,是各占个字节。,11.8,共用体,11.8.2,共用体变量的引用方式,只有先定义了共用体变量才能引用它,而且不,能引用共用体变量,而只能引用共用体变量中的,成员。,例如,:,前面定义了,a,、,b,、,c,为共用体变量,a.i,(引用共用体变量中的整型变量),a.ch,(引用共用体变量中的字符变量),a.f,(引用共用体变量中的实型变量),11.8,共用体,11.8.3,共用体类型数据的特点,(1),同一个内存段可以用来存放几种不同类型的成员,但在每一瞬时只能

50、存放其中一种,而不是同时存放几种。,(2),共用体变量中起作用的成员是最后一次存放的成员,在存入一个新的成员后原有的成员就失去作用。,(3),共用体变量的地址和它的各成员的地址都是同一地址。,11.8,共用体,(4),不能对共用体变量名赋值,也不能企图引用变量名来得到一个值,又不能在定义共用体变量时对它初始化。,(5),不能把共用体变量作为函数参数,也不能使函数带回共用体变量,但可以使用指向共用体变量的指针,(6),共用体类型可以出现在结构体类型定义中,也可以定义共用体数组。反之,结构体也可以出现在共用体类型定义中,数组也可以作为共用体的成员。,11.8,共用体,例,1,12,设有若干个人员的

展开阅读全文
部分上传会员的收益排行 01、路***(¥15400+),02、曲****(¥15300+),
03、wei****016(¥13200+),04、大***流(¥12600+),
05、Fis****915(¥4200+),06、h****i(¥4100+),
07、Q**(¥3400+),08、自******点(¥2400+),
09、h*****x(¥1400+),10、c****e(¥1100+),
11、be*****ha(¥800+),12、13********8(¥800+)。
相似文档                                   自信AI助手自信AI助手
搜索标签

当前位置:首页 > 考试专区 > 中考

移动网页_全站_页脚广告1

关于我们      便捷服务       自信AI       AI导航        获赠5币

©2010-2025 宁波自信网络信息技术有限公司  版权所有

客服电话:4008-655-100  投诉/维权电话:4009-655-100

gongan.png浙公网安备33021202000488号   

icp.png浙ICP备2021020529号-1  |  浙B2-20240490  

关注我们 :gzh.png    weibo.png    LOFTER.png 

客服