《第9章用户建立的数据类型.ppt》由会员分享,可在线阅读,更多相关《第9章用户建立的数据类型.ppt(66页珍藏版)》请在三一办公上搜索。
1、,第九章,用户建立的数据类型,9 用户建立的数据类型,本章要点,结构体的概念结构体的定义和引用结构体数组,9 用户建立的数据类型,主要内容,9.1 结构体类型和结构体变量9.2 结构体数组 9.3 结构体指针 9.4 用指针处理链表9.5 共用体类型9.6 枚举类型 9.7 用typedef命名类型,9 用户建立的数据类型,9.1.1 结构体类型,问题定义:有时需要将不同类型的数据组合成一个有机的整体,以便于引用。如:一个学生有学号/姓名/性别/年龄/地址等属性 int num;char name20;char sex;int age;int score char addr30;,应当把它们组
2、织成一个组合项,在一个组合项中包含若干个类型不同(当然也可以相同)的数据项。,9.1 结构体类型和结构体变量,声明一个结构体类型的一般形式为:struct 结构体名 成员表列;如:struct student int num;char name20;char sex;int age;float score;char addr30;;,结构体名,类型名,成员名,9.1.1 结构体类型,9.1 结构体类型和结构体变量,说明:(1)结构体类型并不是只有一种,而是可以设计出许多种结构体类型.(2)成员也可以是一个结构体变量。(3)“结构体”这个词是根据英文单词译出的。,9.1.1 结构体类型,9.1
3、结构体类型和结构体变量,9.1.2 定义结构体类型变量,可以采取以下3种方法定义结构体类型变量:(1)先声明结构体类型再定义变量名例如:struct student student1,student2;int a,b;定义了student1和student2为struct student类型的变量,即它们具有struct student类型的结构.,9.1 结构体类型和结构体变量,(2)在声明类型的同时定义变量 struct结构体名 成员表列 变量名表列;,9.1.2 定义结构体类型变量,9.1 结构体类型和结构体变量,例如:struct student int num;char name20
4、;char sex;int age;float score;char addr30;student1,student2;,它的作用与第一种方法相同,即定义了两个struct student 类型的变量student1,student2,9.1.2 定义结构体类型变量,9.1 结构体类型和结构体变量,(3)不指定类型名而直接定义结构体类型变量其一般形式为:struct成员表列 变量名表列;即不出现结构体名。,注意:(1)结构体类型与结构体变量是不同的概念,不能混同。(2)结构体类型中的成员名可以与程序中的变量名相同,但二者不代表同一对象。(3)对结构体变量中的成员(即“域”),可以单独使用,它的
5、作用与地位相当于普通变量。,9.1.2 定义结构体类型变量,9.1 结构体类型和结构体变量,9.1.3 引用结构体变量,在定义了结构体变量以后,当然可以引用这个变量。但应遵守以下规则:(1)同类的结构体变量可以互相赋值,如:Student1=student2;(2)不能将一个结构体变量作为一个整体进 行输入和输出。例如:已定义student1和student2为结构体变量并且它们已有值。printf(%d,%s,%c,%d,%f,%n,student1);,9.1 结构体类型和结构体变量,引用结构体变量中成员的方式为结构体变量名.成员名例如:student1.num 表示student1变量中
6、的num成员,即student1的num(学号)项。可以对变量的成员赋值,例如:student1.num=10010;“.”是成员(分量)运算符,它在所有的运算符中优先级最高,因此可以把student1.num作为一个整体来看待。,9.1.3 引用结构体变量,9.1 结构体类型和结构体变量,(2)如果成员本身又属一个结构体类型,则要用若干个成员运算符,一级一级地找到最低的一级的成员。只能对最低级的成员进行赋值或存取以及运算。,9.1.3 引用结构体变量,9.1 结构体类型和结构体变量,Struct date int month;int day;int year;Struct student i
7、nt num20;struct date birthday;student1;,注意:不能用student1.birthday来访问student1变量中的成员birthday,因为birthday本身是一个结构体变量。,9.1.3 引用结构体变量,9.1 结构体类型和结构体变量,Student1.num=2010Student1.Birthday.month=12,(3)对结构体变量的成员可以像普通变量一样进行各种运算(根据其类型决定可以进行的运算)。例如:student2.score=student1.score;sum=student1.score+student2.score;stud
8、ent1.age+;+student2.age;,由于“”运算符的优先级最高,因此是对进行自加运算,而不是先对进行自加运算。,9.1.3 引用结构体变量,9.1 结构体类型和结构体变量,(4)可以引用结构体变量成员的地址,也可以引用结构体变量的地址。例如:scanf(%d,(输入student1.num的值)printf(%o,student1);(输出student1的首地址),9.1.3 引用结构体变量,9.1 结构体类型和结构体变量,但不能用以下语句整体读入结构体变量,例如:scanf(%d,s,c,d,f,s,student1);结构体变量的地址主要用作函数参数,传递结构体变量的地址。
9、,9.1.3 引用结构体变量,9.1 结构体类型和结构体变量,9.1.4 结构体变量的初始化,运行结果:No.:10101name:LiLinsex:address:123 Beijing Road,9.1 结构体类型和结构体变量,例9.1 对结构体变量初始化,9.2 结构体数组,一个结构体变量中可以存放一组数据(如一个学生的学号、姓名、成绩等数据)。如果有个学生的数据需要参加运算,显然应该用数组,这就是结构体数组。结构体数组与以前介绍过的数值型数组不同之处在于每个数组元素都是一个结构体类型的数据,它们都分别包括各个成员(分量)项。,9.2.1 定义结构体数组 和定义结构体变量的方法相仿,只需
10、说明其为数组即可。例如:struct student int num;char name20;char sex;int age;float score;char addr30;struct student stu3;,以上定义了一个数组 stu,数组有个元素,均为 struct student类型数据。,9.2 结构体数组,也可以直接定义一个结构体数组,例如:,9.2 结构体数组,strcut int num;stu 3;,struct student int num;stu 3;,9.2.2 结构体数组的初始化 与其他类型的数组一样,对结构体数组可以初始struct student int
11、num;char name20;char sex;int age;float score;char addr30;stu2 10101,LiLin,M,18,87.5,103 BeijingRoad,10102,Zhang Fun,M,19,99,130 Shanghai Road;,9.2 结构体数组,当然,数组的初始化也可以用以下形式:struct student int num;struct studentstr=,;即先声明结构体类型,然后定义数组为该结构体类型,在定义数组时初始化。,结构体数组初始化的一般形式是在定义数组的后面加上“初值表列;”。,9.2 结构体数组,9.2.3 结构
12、体数组应用举例,9.2 结构体数组,例9.2 对候选人得票的统计程序。设有3个候选人,每次输入一个得票的候选人的名字,要求最后输出各人得票结果。,9.3 结构体指针,结构体指针是指向结构体变量的指针,一个结构体变量的起始地址就是这个结构体变量的指针。指针变量既可以指向结构体变量,也可以用来指向结构体数组中的元素。但是,指针变量的基类型必须与结构体变量的类型相同。例如:struct student*pt;/*pt可以指向struct student类型的数据*/,9.3.1 指向结构体变量的指针变量 下面通过一个简单例子来说明指向结构体变量的指针变量的应用。,运行结果:89101 name:Li
13、Lin sex:score:89.500000:89101 name:LiLin sex:score:89.500000,9.3 结构体指针,例9.3 通过指向结构体变量的指针变量输出该结构体变量的信息。,9.3.1 指向结构体变量的指针变量,程序分析:在函数的执行部分将结构体变量-的起始地址赋给指针变量,也就是使指向-,然后对-的各成员赋值。第一个函数是输出-的各个成员的值。用-表示-中的成员,依此类推。第二个函数也是用来输出-各成员的值,但使用的是(*)这样的形式。,图9-7,9.3 结构体指针,9.3.1 指向结构体变量的指针变量,以下3种形式等价:结构体变量成员名(*)成员名-成员名其
14、中-称为指向运算符。,请分析以下几种运算:-得到指向的结构体变量中的成员的值。-得到指向的结构体变量中的成员的值,用完该值后使它加。-得到指向的结构体变量中的成员的值加,然后再使用它。,9.3 结构体指针,9.3.2 指向结构体数组的指针,运行结果:LiLin 18 Zhang Fun 19 WangMing 20,9.3 结构体指针,例9.4 指向结构体数组元素的指针应用。,9.3.2 指向结构体数组的指针,程序分析:是指向struct student结构体类型数据的指针变量。在for语句中先使的初值为stu,也就是数组stu第一个元素的起始地址。在第一次循环中输出stu0的各个成员值。然后
15、执行,使自加。加意味着p所增加的值为结构体数组stu的一个元素所占的字节数。执行+后p的值等于stu 1,指向stu1。在第二次循环中输出stu1的各成员值。在执行后,p的值等于stu+2,再输出stu 2的各成员值。在执行+后,的值变为stu+,已不再小于stu+3了,不再执行循环。,图9-8,9.3 结构体指针,9.3.2 指向结构体数组的指针,注意:(1)如果的初值为stu,即指向第一个元素,则加后p就指向下一个元素。例如:(+p)-num先使自加,然后得到它指向的元素中的num成员值(即10102)。(p+)-num先得到-num的值(即10101),然后使自加,指向stu1。请注意以
16、上二者的不同。,9.3 结构体指针,9.3.3 结构体变量和指向结构体的指针作函数参数 将一个结构体变量的值传递给另一个函数,有3个方法:用结构体变量的成员作参数。(2)用结构体变量作实参。(3)用指向结构体变量(或数组)的指针作实参,将结构体变量(或数组)的地址传给形参.,9.3 结构体指针,9.3 结构体指针,例9.5 有一个结构体变量stu,内含学生学号、姓名和3门课程的成绩。要求在main函数中赋予值,在另一函数print中将它们输出。今用结构体变量作函数参数。,运行结果:67.50000089.00000078.599998,运行结果:67.50000089.00000078.599
17、998,9.3 结构体指针,例9.6 有一个结构体变量stu,内含学生学号、姓名和3门课程的成绩。要求在main函数中赋予值,在另一函数print中将它们输出。将例题9.5改用指向结构体变量的指针作实参,程序分析:此程序改用在定义结构体变量stu时赋初值,这样程序可简化些。print函数中的形参被定义为指向struct student类型数据的指针变量。注意在调用print函数时,用结构体变量str的起始地址stu作实参。在调用函数时将该地址传送给形参p(p是指针变量)。这样就指向stu。在print函数中输出所指向的结构体变量的各个成员值,它们也就是stu的成员值.main函数中的对各成员赋
18、值也可以改用scanf函数输入.,9.3 结构体指针,9.4.1 链表概述 链表是一种常见的重要的数据结构,是动态地进行存储分配的一种结构。链表的组成:头指针:存放一个地址,该地址指向一个元素 结点:用户需要的实际数据和链接节点的指针,9.4 用指针处理链表,用结构体建立链表:struct student int num;float score;struct student*next;;,9.4 用指针处理链表,成员num和score用来存放结点中的有用数据(用户需要用到的数据),next是指针类型的成员,它指向struct student类型数据(下一个节点),9.4.2 建立简单的静态链表
19、,例9.7 建立一个如图9.11所示的简单链表,它由3个学生数据的结点组成。输出各结点中的数据。,运行结果:1010189.51010390.01010785.0,9.4 用指针处理链表,程序分析:开始时使head指向a结点,a.next指向b结点,b.next指向c结点,这就构成链表关系。“c.next=NULL”的作用是使c.next不指向任何有用的存储单元。在输出链表时要借助p,先使p指向a结点,然后输出a结点中的数据,“p=p-next”是为输出下一个结点作准备。p-next的值是b结点的地址,因此执行“p=p-next”后p就指向b结点,所以在下一次循环时输出的是b结点中的数据。,9
20、.4 用指针处理链表,9.4.3 建立动态链表,9.4 用指针处理链表,所谓建立动态链表是指在程序执行过程中从无到有地建立起一个链表,即一个一个地开辟结点和输入各结点数据,并建立起前后相链的关系。,例9.8 写一函数建立一个有3名学生数据的单向动态链表.,9.4 用指针处理链表,算法的实现:,图9-13,9.4 用指针处理链表,我们约定学号不会为零,如果输入的学号为,则表示建立链表的过程完成,该结点不应连接到链表中。如果输入的p1-num不等于,则输入的是第一个结点数据(n=1),令headp1,即把p1的值赋给head,也就是使head也指向新开辟的结点p1。所指向的新开辟的结点就成为链表中
21、第一个结点,算法的实现:再开辟另一个结点并使p1指向它,接着输入该结点的数据.,如果输入的p1-num,则应链入第个结点(n=2),将新结点的地址赋给第一个结点的next成员.,接着使,也就是使指向刚才建立的结点,图9-14,9.4 用指针处理链表,算法的实现:,在第三次循环中,由于(),又将的值赋给-,也就是将第个结点连接到第个结点之后,并使,使指向最后一个结点.,图9-15,9.4 用指针处理链表,再开辟一个结点并使p1指向它,并输入该结点的数据.,算法的实现:再开辟一个新结点,并使p1指向它,输入该结点的数据。由于p1-num的值为,不再执行循环,此新结点不应被连接到链表中.,将NULL
22、赋给p2-next.,建立链表过程至此结束,p1最后所指的结点未链入链表中,第三个结点的next成员的值为NULL,它不指向任何结点。,图9-16,9.4 用指针处理链表,9.4.4 输出链表 首先要知道链表第一个结点的地址,也就是要知道head的值。然后设一个指针变量p,先指向第一个结点,输出所指的结点,然后使后移一个结点,再输出,直到链表的尾结点。,图9-17,9-18,9.4 用指针处理链表,9.4 用指针处理链表,例9.9 编写一个输出链表的函数print。,例9.9-2 可以把例9.7和例9.9合起来加上一个主函数,组成一个程序。,9.5.1 什么是共用体类型 使几个不同的变量共占同
23、一段内存的结构称为“共用体”类型的结构.,定义共用体类型变量的一般形式为:union共用体名 成员表列变量表列;,图9-19,9.5 共用体类型,例如:union data union data int i;int i;char ch;char ch;float f;float f;a,b,c;union data a,b,c;,9.5 共用体类型,共用体和结构体的比较:结构体变量所占内存长度是各成员占的内存长度之和。每个成员分别占有其自己的内存单元。共用体变量所占的内存长度等于最长的成员的长度。,例如:上面定义的“共用体”变量、各占个字节(因为一个实型变量占个字节),而不是各占个字节。,9.
24、5 共用体类型,9.5.2 共用体变量的引用方式 只有先定义了共用体变量才能引用它,而且不能引用共用体变量,而只能引用共用体变量中的成员。,例如:前面定义了a、b、c为共用体变量 a.i(引用共用体变量中的整型变量)a.ch(引用共用体变量中的字符变量)a.f(引用共用体变量中的实型变量),9.5 共用体类型,9.5.3 共用体类型数据的特点(1)同一个内存段可以用来存放几种不同类型的成员,但在每一瞬时只能存放其中一种,而不是同时存放几种。(2)共用体变量中起作用的成员是最后一次存放的成员,在存入一个新的成员后原有的成员就失去作用。(3)共用体变量的地址和它的各成员的地址都是同一地址。,9.5
25、 共用体类型,(4)不能对共用体变量名赋值,也不能企图引用变量名来得到一个值,又不能在定义共用体变量时对它初始化。(5)以前的C规定不能把共用体变量作为函数参数,但可以使用指向共用体变量的指针作函数参数。ANSI新标准放宽了限制,允许用共用体变量作为函数参数。(6)共用体类型可以出现在结构体类型定义中,也可以定义共用体数组。反之,结构体也可以出现在共用体类型定义中,数组也可以作为共用体的成员。,9.5 共用体类型,图9-10,9.5 共用体类型,例9.10 设有若干个人员的数据,其中有学生和教师。学生的数据中包括:姓名、号码、性别、职业、班级。教师的数据包括:姓名、号码、性别、职业、职务。可以
26、看出,学生和教师所包含的数据是不同的。现要求把它们放在同一表格中。,union categ int banji;char position10;struct int num;char name10;char sex;char job;union categ category;person2;,9.5 共用体类型,也可以不在结构体类型的声明中声明共用体类型,而把它放在结构体类型的声明之前。,枚举:将变量的值一一列举出来,变量的值只限于列举出来的值的范围内。申明枚举类型用enumenum weekdaysun,mon,tue,wed,thu,fri,sat;定义变量:enum weekday wo
27、rkday,week-day;enumsun,mon,tue,wed,thu,fri,satworkday;变量值只能是sun到sat之一,枚举元素枚举常量,9.6 枚举类型,9.6 枚举类型,说明:在编译中,对枚举元素按常量处理,故称枚举常量。它们不是变量,不能对它们赋值。(2)枚举元素作为常量,它们是有值的,语言编译按定义时的顺序使它们的值为,(3)枚举值可以用来作判断比较。(4)一个整数不能直接赋给一个枚举变量。,9.6 枚举类型,例13口袋中有红、黄、蓝、白、黑5种颜色的球若干个。每次从口袋中先后取出个球,问得到3种不同色的球的可能取法,输出每种排列的情况。,算法:图9-22 9-23
28、,13.9 枚举类型,#include main()enum color red,yellow,blue,white,black;enum color i,j,k,pri;int n,loop;n=0;for(i=red;i=black;i+)for(j=red;j=black;j+)if(i!=j)for(k=red;k=black;k+)if(k!=i),13.9 枚举类型,switch(pri)case red:printf(%-10s,red);break;case yellow:printf(%-10s,yellow);break;case blue:printf(%-10s,blu
29、e);break;case white:printf(%-10s,white);break;case black:printf(%-10s,black);break;default:break;printf(n);printf(ntotal:%5dn,n);,运行情况如下:1redyellowblue2redyellowwhite3redyellowblack58blackwhitered59blackwhiteyellow60blackwhiteblue total:60,9.7 用typedef命名类型,用typedef声明新的类型名来代替已有的类型名声明INTEGER为整型typedef
30、 int INTEGER声明结构类型Typedef struct int month;int day;int year;DATE;,9.7 用typedef命名类型,声明为整型数组类型;声明为字符指针类型 typedef char*STRING;声明POINTER为指向函数的指针类型,该函数返回整型值 typedef int(*POINTER)(),9.7 用typedef命名类型,用typedef定义类型的方法 先按定义变量的方法写出定义体(如:int i)。将变量名换成新类型名(例如:将i换成COUNT)。在最前面加(例如:typedef int COUNT)。然后可以用新类型名去定义变量
31、。,9.7 用typedef命名类型,用typedef定义类型的方法(举例)先按定义数组变量形式书写:int n100;将变量名换成自己指定的类型名:intNUM0;在前面加上typedef,得到 typedef int NUM;用来定义变量:NUM;,9.7 用typedef命名类型,说明:用typedef可以声明各种类型名,但不能用来定义变量。(2)用typedef只是对已经存在的类型增加一个类型名,而没有创造新的类型。(3)当不同源文件中用到同一类型数据时,常用typedef声明一些数据类型,把它们单独放在一个文件中,然后在需要用到它们的文件中用#include命令把它们包含进来。(4)使用typedef有利于程序的通用与移植。,9.7 用typedef命名类型,说明:(5)typedef与#define有相似之处,例如:typedef int COUNT;#define COUNT int的作用都是用COUNT代表int。但事实上,它们二者是不同的。#define是在预编译时处理的,它只能作简单的字符串替换,而typedef是在编译时处理的。实际上它并不是作简单的字符串替换,而是采用如同定义变量的方法那样来声明一个类型,