版權(quán)說(shuō)明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡(jiǎn)介
第7章結(jié)構(gòu)體與鏈表7.1結(jié)構(gòu)體的引出7.2結(jié)構(gòu)體變量7.3結(jié)構(gòu)體數(shù)組7.4結(jié)構(gòu)體類型的指針變量7.5結(jié)構(gòu)體與函數(shù)7.6鏈表在實(shí)際問(wèn)題中我們常需要把不同類型的幾個(gè)數(shù)據(jù)組合起來(lái),構(gòu)成一個(gè)整體,如學(xué)校中教師和學(xué)生的信息。學(xué)號(hào)姓名性別年齡入學(xué)成績(jī)所屬學(xué)院0501李明男19610信息0502張莉女19595信息0503王濤男20580控制職工編號(hào)姓名性別民族出生日期職稱學(xué)歷單位工齡1997025孫杰男漢1974.9講師大學(xué)信息92001016趙玫女回1978.3助教大學(xué)控制51985104鄭毅男漢1964.11教授大學(xué)管理217.1結(jié)構(gòu)體的引出學(xué)號(hào)姓名性別年齡入學(xué)成績(jī)所屬學(xué)院0501李明男19610信息0502張莉女19595信息0503王濤男20580控制如何表示這樣的數(shù)據(jù)信息?結(jié)構(gòu)體是由一些邏輯相關(guān),但數(shù)據(jù)類型不同的分量組成的一組數(shù)據(jù)。學(xué)號(hào)姓名性別年齡入學(xué)成績(jī)所屬學(xué)院intnumcharname[10]charsexintageintscorecharinstitute[20]例:輸入三個(gè)學(xué)生的信息,并輸出#include<stdio.h>voidmain(){structstudent{intnum;charname[10];charsex;
intage;
intscore;charinstitute[20]};
structstudents1,s2,s3;
定義學(xué)生的結(jié)構(gòu)體類型定義3個(gè)結(jié)構(gòu)體類型的變量,用來(lái)存放3個(gè)學(xué)生的信息7.1結(jié)構(gòu)體的引出structstudent{intnum;charname[10];charsex;
intage;
intscore;charinstitute[20]};7.2.1結(jié)構(gòu)體的定義注意:定義了結(jié)構(gòu)體類型,僅僅是定義了數(shù)據(jù)的組織形式,創(chuàng)立了一種數(shù)據(jù)類型,但并不會(huì)為這種結(jié)構(gòu)體類型分配內(nèi)存空間只有定義了結(jié)構(gòu)體變量,才會(huì)為變量分配空間注意不要忘了分號(hào)成員表列struct
結(jié)構(gòu)體類型名{數(shù)據(jù)類型成員名1;
數(shù)據(jù)類型成員名2;::
數(shù)據(jù)類型成員名n;}
;關(guān)鍵字用戶定義的標(biāo)識(shí)符7.2結(jié)構(gòu)體變量定義結(jié)構(gòu)體變量的方法(1)先定義結(jié)構(gòu)體類型,再定義變量
structstudent{charname[10];
intage;
ints1,s2;};
structstudentst1,st2;st1st2nameages1s2nameages1s2內(nèi)存中結(jié)構(gòu)體變量占有一片連續(xù)的存儲(chǔ)單元,其占用的字節(jié)數(shù)可用sizeof
運(yùn)算符算出:
printf(“%d\n”,sizeof(structstudent));
printf(“%d\n”,sizeof(st1));結(jié)構(gòu)體變量st1和st2各自都需要22個(gè)字節(jié)的存儲(chǔ)空間結(jié)構(gòu)體類型定義結(jié)構(gòu)體變量定義7.2結(jié)構(gòu)體變量定義結(jié)構(gòu)體變量的方法(2)定義結(jié)構(gòu)體類型同時(shí)定義變量
structstudent{charname[10];
intage;
ints1,s2;}
st1,st2;(3)直接定義結(jié)構(gòu)體變量
struct
{charname[10];
intage;
ints1,s2;}
st1,st2;注意:這里沒有結(jié)構(gòu)體類型名這種方式有時(shí)使用并不方便因此不建議大家采用定義結(jié)構(gòu)體變量7.2結(jié)構(gòu)體變量例:
structdate{intyear;
intmonth;
intday;};
structstud{charname[10];
structdatebirthday;
ints1,s2;};結(jié)構(gòu)體類型可以嵌套定義
或:structstud{charname[10];
structdate{intyear;
intmonth;
intday;}birthday;
ints1,s2;};7.2結(jié)構(gòu)體變量7.2.2結(jié)構(gòu)體變量的引用和初始化格式:
結(jié)構(gòu)體變量名.
成員名structstudent{charname[10];
intage;
ints1,s2;};structstudentst1;st1.
name
=“Mary”;st1.age
=21;st1.
s1
=78;st1.
s2
=86;說(shuō)明:一般情況下都是對(duì)結(jié)構(gòu)體變量的成員進(jìn)行賦值和輸入\輸出7.2結(jié)構(gòu)體變量structdate{intyear;
intmonth;
intday;};structstud{charname[10];
intage;
structdatebirthday;
ints1,s2;};structstudst2;intage,year;=“John”;st2.age=20;st2.birthday.year=1980;st2.birthday.month=11;st2.birthday.day=23;st2.s1=89;st2.s2=95;age=24;year=2000;可以定義與結(jié)構(gòu)體成員名相同名字的變量,它們之間不會(huì)發(fā)生混亂相同類型的結(jié)構(gòu)體變量可以進(jìn)行整體賦值
structdate{intyear;
intmonth;
intday;};structstud{charname[10];
intage;
structdatebirthday;
ints1,s2;};structstudst1,st2,st3;=“John”;st1.age=20;st1.birthday.year=1980;st1.birthday.month=11;st1.birthday.day=23;st1.s1=89;st1.s2=95;st2=st1;=“Mary”;st3.age=20;st3.birthday=st1.birthday;st3.s1=76;st3.s2=85;注意要正確賦值的條件是變量st1已經(jīng)有了數(shù)據(jù)7.2結(jié)構(gòu)體變量structstudent{charname[10];
intage;
ints1,s2;}
st1={“Mary”,21,78,86};structstud{charname[10];
structdatebirthday;
ints1,s2;};structstudst2={“John”,
1980,11,23
,89,95};structstudent{charname[10];
intage;
ints1,s2;};structstudentst1;
st1={“Mary”,21,78,86};初始化,正確這是賦值,錯(cuò)誤C不允許這么做初始化,正確7.2.2結(jié)構(gòu)體變量的引用和初始化結(jié)構(gòu)體變量的輸入輸出
C語(yǔ)言不允許結(jié)構(gòu)體變量整體進(jìn)行輸入和輸出,
只能對(duì)結(jié)構(gòu)體變量的成員進(jìn)行輸入和輸出gets(
);scanf(“%d%d%d”,
&st1.birthday.year
,
&st1.birthday.month,
&st1.birthday.day);scanf(“%d%d%d”,
&st1.age,
&st1.s1,
&st1.s2);
puts();printf(“%4d”,st1.age);printf(“%d.%d.%d”,st1.birthday.year,st1.birthday.month,st1.birthday.day);printf(“%4d%4d\n”,st1.s1,st1.s2);7.2.2結(jié)構(gòu)體變量的引用和初始化7.3結(jié)構(gòu)體數(shù)組學(xué)號(hào)姓名性別年齡入學(xué)成績(jī)所屬學(xué)院0501李明男19610信息0502張莉女19595信息0503王濤男20580控制7.3.1結(jié)構(gòu)體數(shù)組的定義一個(gè)結(jié)構(gòu)體變量只能存放一個(gè)學(xué)生的信息,對(duì)于多個(gè)學(xué)生的信息,可以使用一個(gè)結(jié)構(gòu)體數(shù)組來(lái)存放,結(jié)構(gòu)體數(shù)組的每個(gè)元素是一個(gè)結(jié)構(gòu)體類型的變量定義結(jié)構(gòu)體數(shù)組的方法與定義普通數(shù)組的方法類似:結(jié)構(gòu)體類型數(shù)組名[數(shù)組的長(zhǎng)度];例:輸入30個(gè)學(xué)生的信息,并輸出#include<stdio.h>voidmain(){structstudent{intnum;charname[10];charsex;
intage;
intscore;charinstitute[20]};
structstudents[30];
inti;
for(i=0;i<30;i++){scanf(“%c%d%d%d”,&s[i].sex,&s[i].num,&s[i].age,&s[i].score);gets(s[i].name);gets(s[i].institute);}for(i=0;i<30;i++){printf(“%6d%10s%2c%3d”,s[i].num,s[i].name,s[i].sex,s[i].age);
printf(“%5d%20s\n”,s[i].score,s[i].institue);}}7.3結(jié)構(gòu)體數(shù)組7.3.1結(jié)構(gòu)體數(shù)組的定義1、定義結(jié)構(gòu)體數(shù)組(1)先定義結(jié)構(gòu)體類型再定義結(jié)構(gòu)體數(shù)組
structstudent{charname[10];
intage;
ints1,s2;};structstudentst[6];(2)定義結(jié)構(gòu)體類型的同時(shí)定義結(jié)構(gòu)體數(shù)組
structstudent
{charname[10];
intage;
ints1,s2;}
st[6];(3)直接定義結(jié)構(gòu)體數(shù)組
struct
{charname[10];
intage;
ints1,s2;}
st[6];不提倡使用該方法7.3.2結(jié)構(gòu)體數(shù)組的初始化將每個(gè)數(shù)組元素的數(shù)據(jù)用花括號(hào){}括起來(lái)structstudent{charname[10];
intage;
ints1,s2;};structstudentst[3]={
{“Mary”,21,78,86},
{“Alex”,20,90,80},
{“Mike”,19,75,68}
};Mary217886Alex209080Mike197568st[0]st[1]st[2]7.3結(jié)構(gòu)體數(shù)組(2)數(shù)組元素之間可以整體賦值也可以將一個(gè)元素賦給一個(gè)相同類型的結(jié)構(gòu)體變量structstudent
st[3]={
{“Mary”,21,78,86},{“Alex”,…}
};structstudent
x;st[2]=st[0];x=st[1];都是結(jié)構(gòu)體變量的整體賦值形式7.3.3結(jié)構(gòu)體數(shù)組的使用(1)引用某個(gè)數(shù)組元素的成員
例:puts(
st[0].name
);
printf(“%d,%d”,
st[1].age,st[1].s1
);(3)輸入和輸出操作只能對(duì)數(shù)組元素的成員進(jìn)行7.3結(jié)構(gòu)體數(shù)組分析:假設(shè)有3個(gè)候選人,共有100個(gè)人投票定義一個(gè)結(jié)構(gòu)體數(shù)組,它有3個(gè)元素代表3個(gè)候選人,每個(gè)元素有2個(gè)成員,一個(gè)是候選人名字,一個(gè)是得票數(shù)(初始時(shí)為0)例:設(shè)計(jì)一個(gè)對(duì)候選人得票進(jìn)行統(tǒng)計(jì)的程序候選人票數(shù)Mike0John0Alex0有100張選票,輸入選票上的名字,然后判斷是誰(shuí)的名字,就將誰(shuí)的票數(shù)加1,重復(fù)100次最后輸出結(jié)構(gòu)體數(shù)組#include<stdio.h>#include<string.h>structperson{charname[10];
intcount;};voidmain(){structperson
cand[3]={{“Li”,0},{“Zhang”,0},{“Fun”,0}};
inti,j;charcname[20];for(i=0;i<100;i++){scanf(“%s”,cname);for(j=0;j<3;j++)if(strcmp(cname,cand[j].name)==0)cand[j].count++;
}for(i=0;i<3;i++)
printf(“%10s:%d\n”,cand[i].name,cand[i].count);}定義候選人的結(jié)構(gòu)體類型對(duì)結(jié)構(gòu)體數(shù)組進(jìn)行初始化//輸入選票上的名字//若有100人投票,則循環(huán)100次//將選票上的名字依次和候選人的名字比較//選票上名字和某個(gè)候選人名字相同時(shí),其票數(shù)加1例:按成績(jī)對(duì)學(xué)生信息進(jìn)行從高到底的排序#include<stdio.h>#defineN30structstud{intn;//學(xué)生學(xué)號(hào)charname[10];//學(xué)生姓名
ints;//學(xué)生成績(jī)};voidinput(structstud
a[]){inti;for(i=0;i<N;i++)
scanf(“%d%s%d”,&a[i].n,a[i].name,&a[i].s);
}voidoutput(structstud
a[
]){inti;for(i=0;i<N;i++)
printf(“%4d%10s%4d”,a[i].n,a[i].name,a[i].s);
}
注意a[i].name前不加&,因name是數(shù)組名,因用%s,輸入時(shí)名字不能加空格voidsort(structstud
a[]
){inti,j;
structstud
temp;
for(i=0;i<N-1;i++)for(j=i+1;j<N;j++)
if(a[i].s<a[j].s)
{temp=a[i];a[i]=a[j];a[j]=temp;}}voidmain(){
structstud
st[N];
input(st);sort(st);output(st);}注意進(jìn)行比較的是元素a[i]和a[j]的成績(jī)成員s,但進(jìn)行交換的是元素a[i]和a[j]例:用冒泡排序法對(duì)6個(gè)數(shù)進(jìn)行排序(從小到大)
9
7
2
5
4
1a[0]a[1]a[2]a[3]a[4]a[5]
7
2
5
4
1
92775471
2
5
4
1
7
94515
2
4
1
5
7
9
2
1
4
5
7
91412冒泡排序方法:依次比較相鄰的兩個(gè)數(shù),將小數(shù)放前面,大數(shù)放后面.n個(gè)數(shù)排序需要進(jìn)行n-1輪比較,從第1輪到第n-1輪,各輪的比較次數(shù)依次為:n-1次、n-2次…1次
9
7
2
5
4
19999972541初始狀態(tài)第1輪第2輪第3輪第4輪第5輪74.1一維數(shù)組冒泡排序#include<stdio.h>#defineN6voidmain(){int
a[N],i,j,t;
for(i=0;i<N;i++)
scanf(“%d”,&a[i]);
for(i=0;i<N-1;i++)for(j=0;j<N-1-i;j++)if(a[j]>a[j+1]){t=a[j];
a[j]=a[j+1];a[j+1]=t;
}for(i=0;i<N;i++)
printf(“%3d”,a[i]);}冒泡排序的改進(jìn)方法
#include<stdio.h>voidmain(){inta[6],i,j,t,flag;for(i=0;i<6;i++)
scanf(“%d”,&a[i]);
i=0;
do{flag=0;for(j=0;j<5-i;j++)if(a[j]>a[j+1]){t=a[j];a[j]=a[j+1];a[j+1]=t;flag=1;
}
i++;
}while(flag);for(i=0;i<6;i++)
printf(“%3d”,a[i]);}例:用選擇排序法對(duì)6個(gè)數(shù)進(jìn)行排序(從小到大)
9
7
2
5
4
1a[0]a[1]a[2]a[3]a[4]a[5]選擇排序方法:第1輪比較時(shí),用a[0]依次與a[1]到a[5]進(jìn)行比較,如果a[0]較大則進(jìn)行交換,第1輪結(jié)束后,a[0]中為最小數(shù).以后各輪比較過(guò)程與第1論類似.
1
9
7
5
4
2
1
2
9
7
5
479574524
1
2
4
9
7
5
1
2
4
5
9
7795745
9
7
2
5
4
1729712初始狀態(tài)第1輪第2輪第3輪第4輪第5輪7957794.1一維數(shù)組選擇排序#include<stdio.h>#defineN6voidmain(){int
a[N],i,j,t;for(i=0;i<N;i++)
scanf(“%d”,&a[i]);
for(i=0;i<N-1;i++)for(j=i+1;j<N;j++)if(a[i]>a[j]){t=a[i];
a[i]=a[j];
a[j]=t;
}for(i=0;i<N;i++)
printf(“%3d”,a[i]);}選擇排序改進(jìn)方法#include<stdio.h>voidmain(){inta[6],i,j,k,t;for(i=0;i<6;i++)
scanf(“%d”,&a[i]);for(i=0;i<5;i++){k=i;for(j=i+1;j<6;j++)if(a[k]>a[j])k=j;if(k!=i){t=a[i];a[i]=a[k];
a[k]=t;}//endif}//endforfor(i=0;i<6;i++)
printf(“%3d”,a[i]);}7.4結(jié)構(gòu)體類型的指針變量
7.4.1指向結(jié)構(gòu)體變量的指針
1.定義
structstudent
{charname[20];
intage;
ints1,s2;};
structstudent
stu,*p;
p=&stu;2.成員的引用格式(1)結(jié)構(gòu)體變量名.成員名
gets(
);(*p).age
=21;p->s1
=87;p->s2
=90;(2)(*指針變量名).成員名
(*p).age(3)指針變量名->成員名
p
->s1typedef
structstudent{charname[20];
intage;
ints1,s2;}SD;
SDx,stu,*p;
p=&stu;成員的引用格式(1)結(jié)構(gòu)體變量名.成員名例:scanf(“%s”,);scanf(“%d”,&x.age);x.s1=89;x.s2=78;gets();(*p).age=21;scanf(“%d”,&p->s1);p->s2=90;(2)(*指針變量名).成員名(*p).age(3)指針變量名->成員名p->s17.4結(jié)構(gòu)體與指針說(shuō)明:和成員運(yùn)算符一樣,“->”為指向運(yùn)算符,是運(yùn)算優(yōu)先級(jí)最高的運(yùn)算符。由于成員運(yùn)算符“.”的運(yùn)算優(yōu)先級(jí)高于運(yùn)算符“*”,
因此(*p).成員名中()不能少。*p.成員名p=&stu.score;
不能用指向某個(gè)結(jié)構(gòu)體變量的指針指向該結(jié)構(gòu)體變量的某個(gè)成員。7.4.2指向結(jié)構(gòu)體數(shù)組的指針1.定義structstudent
a[3],*p
;2.使用for(
p=a;p<a+3;p++
){gets(
p->name);
scanf(“%d%d%d”,&p->age,&p->s1,&p->s2);}
7.4結(jié)構(gòu)體與指針例如:
struct
struct_name{charname[10];
intnum;floatscore;};/*定義結(jié)構(gòu)體類型標(biāo)識(shí)符*/
struct
struct_namestd[30],*p;900390趙明879002王建899001李紅結(jié)構(gòu)體數(shù)組stdstd[0]std[1]std[2]p賦值語(yǔ)句p=std;/*p指向一個(gè)結(jié)構(gòu)體數(shù)組std*//*指針變量p存放的是數(shù)組std的首地址*/引用:p->name;p->num;p->score;趙明909003879002王建899001李紅結(jié)構(gòu)體數(shù)組stdstd[0]std[1]std[2]思考:賦值語(yǔ)句
p=std+1;和p++;分別代表指針p指向哪里?p->name;p->num;p->score;代表的信息發(fā)生了什么變化?pp注意:以下賦值語(yǔ)句都是錯(cuò)誤的:p=&std[1].score;(不能指向數(shù)組元素的成員變量)p=&std;(數(shù)組名本身就代表該數(shù)組的首地址,因此不能使用地址運(yùn)算符&)7.5.1結(jié)構(gòu)體變量作為函數(shù)參數(shù)1.函數(shù)實(shí)參和形參都用結(jié)構(gòu)體變量,參數(shù)之間為值傳遞,實(shí)參結(jié)構(gòu)體變量各成員的值依次傳給形參結(jié)構(gòu)體變量2.返回結(jié)構(gòu)體類型值的函數(shù)定義格式:結(jié)構(gòu)體類型名函數(shù)名(形參表){函數(shù)體;}
例:structstudentfunct(intx,floaty){函數(shù)體;}注意:結(jié)構(gòu)體類型是已經(jīng)定義好的7.5結(jié)構(gòu)體與函數(shù)#include<stdio.h>#defineN5structstud{charname[10];
ints[3];
intsum,ave;};structstud
count(structstudx){intj;
x.sum=0;for(j=0;j<3;j++)
x.sum=x.sum+x.s[j];
x.ave=x.sum/3;
return(x);}例:求學(xué)生成績(jī)的總分和平均分(結(jié)構(gòu)體變量作參數(shù))//定義學(xué)生的結(jié)構(gòu)體類型//計(jì)算學(xué)生三門課的總分//返回學(xué)生的全部信息//結(jié)構(gòu)體變量作參數(shù)返回結(jié)構(gòu)體類型的值voidmain(){structstuda[N];
intj;for(j=0;j<N;j++)
scanf(“%s%d%d%d”,a[j].name,&a[j].s[0],
&a[j].s[1],&a[j].s[2]);
for(j=0;j<N;j++)
a[j]=count(a[j]);
for(j=0;j<N;j++)printf(“%10s%4d%4d%4d%6d%4d\n”,a[j].name,a[j].s[0],a[j].s[1],a[j].s[2],a[j].sum,a[j].ave);}//函數(shù)調(diào)用,將a[j]的值傳給count函數(shù)的參數(shù)x,并將返回值賦給a[j]//輸入N個(gè)學(xué)生的信息//定義結(jié)構(gòu)體數(shù)組a,存放N個(gè)學(xué)生的信息void
count(structstud*p){intj;p->sum=0;for(j=0;j<3;j++)p->sum=p->sum+p->s[j];p->ave=p->sum/3;}例:求學(xué)生成績(jī)的總分和平均分(指向結(jié)構(gòu)體的指針作參數(shù))#include<stdio.h>#defineN5structstud{charname[10];
ints[3];
intsum,ave;};//指向結(jié)構(gòu)體的指針作參數(shù)返回結(jié)構(gòu)體類型的值voidmain(){structstuda[N];
intj;for(j=0;j<N;j++)
scanf(“%s%d%d%d”,a[j].name,&a[j].s[0],
&a[j].s[1],&a[j].s[2]);
for(j=0;j<N;j++)
count(&a[j]);
for(j=0;j<N;j++)printf(“%10s%4d%4d%4d%6d%4d\n”,a[j].name,a[j].s[0],a[j].s[1],a[j].s[2],a[j].sum,a[j].ave);}voidmain(){structstuda[N],*q;
intj;
q=a;for(j=0;j<N;j++)
scanf(“%s%d%d%d”,q->name,&a[j].s[0],
&a[j].s[1],&a[j].s[2]);
for(j=0;q<a+N;q++)
count(q);
for(j=0;j<N;j++)printf(“%10s%4d%4d%4d%6d%4d\n”,a[j].name,a[j].s[0],a[j].s[1],a[j].s[2],a[j].sum,a[j].ave);}7.6鏈表鏈表:是可以動(dòng)態(tài)地進(jìn)行存儲(chǔ)分配的一種數(shù)據(jù)結(jié)構(gòu)它是由一組動(dòng)態(tài)數(shù)據(jù)鏈接而成的序列結(jié)點(diǎn):鏈表中的每一個(gè)動(dòng)態(tài)數(shù)據(jù)稱為一個(gè)結(jié)點(diǎn)表頭結(jié)點(diǎn)表尾結(jié)點(diǎn)NULL為空地址
表示鏈表到此結(jié)束2010142815709514281861570282NULL3中間結(jié)點(diǎn)一.、基本概念1.動(dòng)態(tài)存儲(chǔ)分配:根據(jù)需要臨時(shí)分配內(nèi)存單元用以存放數(shù)據(jù),
當(dāng)數(shù)據(jù)不用時(shí)可以隨時(shí)釋放內(nèi)存單元2.鏈表:是可以動(dòng)態(tài)地進(jìn)行存儲(chǔ)分配的一種數(shù)據(jù)結(jié)構(gòu)它是由一組動(dòng)態(tài)數(shù)據(jù)鏈接而成的序列3.結(jié)點(diǎn):鏈表中的每一個(gè)動(dòng)態(tài)數(shù)據(jù)稱為一個(gè)結(jié)點(diǎn)4.結(jié)點(diǎn)類型:是一個(gè)包含指針項(xiàng)的結(jié)構(gòu)體類型一般由兩部分組成:(1)數(shù)據(jù)成員:存放數(shù)據(jù)(2)指針成員:存放下一個(gè)結(jié)點(diǎn)的地址struct
sd{intnum;
intscore;
struct
sd*next;};
數(shù)據(jù)成員指針成員鏈表的基本概念2010142815709514281861570282NULL3鏈表的基本概念2010head2010142815709514281861570282NULL3鏈表的每個(gè)結(jié)點(diǎn)存放在內(nèi)存中的不同位置,只有找到第1個(gè)結(jié)點(diǎn),才能通過(guò)第1個(gè)結(jié)點(diǎn)的指針成員找到第2個(gè)結(jié)點(diǎn)…因此我們將第1個(gè)結(jié)點(diǎn)的地址存放在頭指針中頭指針鏈表的長(zhǎng)度是不固定的,可以隨時(shí)添加結(jié)點(diǎn),如果將一個(gè)結(jié)點(diǎn)添加到鏈表的尾部,則新結(jié)點(diǎn)成為表尾結(jié)點(diǎn),它的指針成員必須賦為NULL,而原來(lái)的表尾結(jié)點(diǎn)則成為中間結(jié)點(diǎn)75NULL436923692靜態(tài)簡(jiǎn)單鏈表#include<stdio.h>typedef
structstud{intnum,score;
structstud*next;}SD;head2010142815702010abc19514282861570382NULLp201014281570NULLvoidmain(){SD
a,b,c,*head,*p;
head=&a;a.num=1;a.score=95;a.next=&b;b.num=2;b.score=86;b.next=&c;c.num=3;c.score=82;c.next=NULL;
p=head;while(p!=NULL){printf(“%3d%4d\n”,p->num,p->score);
p=p->next;}}靜態(tài)簡(jiǎn)單鏈表#include<stdio.h>typedef
structstud{intnum,score;
structstud*next;}SD;voidmain(){SDa,b,c,*head,*p;head=&a;
a.num=1;a.score=95;a.next=&b;
b.num=2;b.score=86;b.next=&c;
c.num=3;c.score=82;c.next=NULL;
while(head!=NULL){printf("%3d%4d\n",head->num,head->score);head=head->next;}}//也可以實(shí)現(xiàn)上一頁(yè)的功能動(dòng)態(tài)鏈表的建立建立鏈表的方法表尾添加法:新結(jié)點(diǎn)作為表尾結(jié)點(diǎn)表首添加法:新結(jié)點(diǎn)作為表頭結(jié)點(diǎn)95NULL1201086NULL21428201082NULL3157095NULL1201086NULL2142882NULL31570head201014281570head20101428處理動(dòng)態(tài)鏈表所需的函數(shù)(需用頭文件<stdlib.h>)1.malloc
函數(shù)原型:void*malloc(unsignedintsize)
作用:在內(nèi)存中開辟一個(gè)長(zhǎng)度為size的連續(xù)存儲(chǔ)空間,
并將此存儲(chǔ)空間的起始地址帶回注意:(1)函數(shù)返回值是指針,但該指針是指向void類型的,因此在使用時(shí)希望這個(gè)指針指向其他類型需要用強(qiáng)制類型轉(zhuǎn)換(2)如果內(nèi)存缺少足夠大的空間進(jìn)行分配,則malloc
函數(shù)返回值為“空指針”(即NULL)例:struct
sd*p;p=
(struct
sd*)
malloc(
sizeof(struct
sd)
);強(qiáng)制類型轉(zhuǎn)換結(jié)構(gòu)體類型占用的字節(jié)長(zhǎng)度2.free函數(shù)原型:voidfree(void*ptr)
作用:將指針變量ptr指向的存儲(chǔ)空間釋放注意:ptr的值不是任意的地址,必須是程序中執(zhí)行malloc
函數(shù)所返回的地址(2)模型中ptr是void型,但調(diào)用free函數(shù)時(shí),參數(shù)可能是其他類型,計(jì)算機(jī)系統(tǒng)會(huì)自動(dòng)進(jìn)行轉(zhuǎn)換
例:struct
sd*p;p=(struct
sd*)malloc(sizeof(struct
sd));free(p);p42004200動(dòng)態(tài)鏈表的建立表尾添加法建立鏈表#include<stdio.h>#include<stdlib.h>typedef
structstudent{intnum;
intscore;
structstudent*next;}ST;#defineLENsizeof(ST)intn;定義ST,以后書寫簡(jiǎn)單求出結(jié)構(gòu)體類型占用的字節(jié)數(shù)全局量n表示結(jié)點(diǎn)的個(gè)數(shù)ST*creat(void){ST*p1,*p2,*head=NULL;n=0;
p1=(ST*)malloc(LEN);if(p1==NULL){printf("\nNoenoughmemory!\n");exit(0);}
scanf("%d%d",&p1->num,&p1->score);while(p1->num!=0){n=n+1;if(n==1)head=p1;elsep2->next=p1;p2=p1;
p1=(ST*)malloc(LEN);if(p1==NULL){printf("\nNoenoughmemory!\n");exit(0);}
scanf("%d%d",&p1->num,&p1->score);}p2->next=NULL;
free(p1);return(head);}//產(chǎn)生一個(gè)新結(jié)點(diǎn)//輸入新結(jié)點(diǎn)的數(shù)據(jù)成員//結(jié)點(diǎn)個(gè)數(shù)加1//讓指針p2指向p1所指向的結(jié)點(diǎn)//表尾結(jié)點(diǎn)的指針成員賦為空//釋放p1所指向的結(jié)點(diǎn)空間//p1指向新產(chǎn)生的結(jié)點(diǎn)//若n等于1,則p1當(dāng)前所指向的是表頭結(jié)點(diǎn)由于head頭指針不變,所以讓下一個(gè)結(jié)點(diǎn)p2記錄表頭結(jié)點(diǎn)的地址,讓表頭結(jié)點(diǎn)去產(chǎn)生一個(gè)新結(jié)點(diǎn),這樣再讓p2->next指向新結(jié)點(diǎn),即可產(chǎn)生鏈接表尾添加法建立鏈表的過(guò)程演示ST*creat(void){ST*p1,*p2,
*head;head=NULL;n=0;p1=(ST*)malloc(LEN);
scanf("%d%d",&p1->num,&p1->score);while(p1->num!=0){n=n+1;if(n==1)head=p1;elsep2->next=p1;p2=p1;p1=(ST*)malloc(LEN);
scanf("%d%d",&p1->num,&p1->score);}p2->next=NULL;free(p1);return(head);}head201014281570NULLp1p23264201020101951428n201014282863820014281570NULL1570157032640123動(dòng)態(tài)鏈表的遍歷(輸出)輸出鏈表voidlist(ST*head){ST*p;
p=head;while(p!=NULL){printf(“%d,%d\n”,p->num,p->score);
p=p->next;
}}head201014281570201019514282861570382NULLp201014281570NULL輸出:1,952,863,82voidmain(){ST*h=NULL;
h=creat();list(h);}951428120108615702142882NULL315702010head鏈表的刪除結(jié)點(diǎn)操作刪除表頭結(jié)點(diǎn)讓頭指針指向鏈表的第2個(gè)結(jié)點(diǎn)1428step1:讓指針變量p指向要?jiǎng)h除的結(jié)點(diǎn)即表頭結(jié)點(diǎn)step2:重新給頭指針賦值,使它指向第2個(gè)結(jié)點(diǎn)head=p->next;step3:釋放刪除的結(jié)點(diǎn)空間free(p);p=head;2010p應(yīng)為要釋放刪除的結(jié)點(diǎn),所以必須在知道此結(jié)點(diǎn)的地址的情況下,才能進(jìn)行空間釋放,所以讓p記錄此結(jié)點(diǎn)的地址。951428120108615702142882NULL315702010head鏈表的刪除結(jié)點(diǎn)操作刪除表尾結(jié)點(diǎn)將鏈表的倒數(shù)第2個(gè)結(jié)點(diǎn)的指針成員賦為NULLstep1:讓指針變量p指向要?jiǎng)h除的結(jié)點(diǎn)即表尾結(jié)點(diǎn)
讓指針變量q指向要?jiǎng)h除結(jié)點(diǎn)的前一個(gè)結(jié)點(diǎn)1570p1428qstep2:將刪除結(jié)點(diǎn)的前驅(qū)結(jié)點(diǎn)的指針成員賦空值step3:釋放刪除的結(jié)點(diǎn)空間free(p);q->next=NULL;NULL951428120108615702142882NULL315702010head鏈表的刪除結(jié)點(diǎn)操作刪除中間結(jié)點(diǎn)step1:讓p指向要?jiǎng)h除的結(jié)點(diǎn),
讓q指向要?jiǎng)h除結(jié)點(diǎn)的前驅(qū)結(jié)點(diǎn)1428p2010qstep2:前驅(qū)結(jié)點(diǎn)的指針成員賦為要?jiǎng)h除結(jié)點(diǎn)的指針成員值step3:釋放刪除的結(jié)點(diǎn)空間free(p);q->next=p->next;讓要?jiǎng)h除結(jié)點(diǎn)的前驅(qū)結(jié)點(diǎn)指向要?jiǎng)h除結(jié)點(diǎn)的后繼結(jié)點(diǎn)1570ST*del(ST*head,intnum){ST*p,*q=NULL;p=head;
while((num!=p->num)&&(p->next!=NULL)){q=p;p=p->next;}if(num==p->num){if(p==head)head=p->next;elseq->next=p->next;
free(p);
n=n-1;
printf(“deleted!\n”);}elseprintf(“cannotdelete!\n”);
return(head);}typedef
structstudent{intnum;
intscore;
structstudent*next;}ST;鏈表的刪除結(jié)點(diǎn)操作刪除結(jié)點(diǎn)函數(shù)令p指向要?jiǎng)h除的結(jié)點(diǎn),q指向其前驅(qū)結(jié)點(diǎn)(方法)//刪除結(jié)點(diǎn)為表頭結(jié)點(diǎn)//刪除結(jié)點(diǎn)為表尾結(jié)點(diǎn)或中間結(jié)點(diǎn)//釋放已刪除的結(jié)點(diǎn)空間//鏈表的結(jié)點(diǎn)個(gè)數(shù)減1//返回鏈表的頭指針要求:刪除與num相等的結(jié)點(diǎn)ST*p,*q=NULL;
p=head;while((num!=p->num)&&(p->next!=NULL)){q=p;p=p->next;}951428120108615702142882NULL315702010head2010pNULLq設(shè)num=32010142814281570讓p指向要?jiǎng)h除的結(jié)點(diǎn),讓q指向其前驅(qū)結(jié)點(diǎn)鏈表的刪除結(jié)點(diǎn)操作鏈表的插入結(jié)點(diǎn)操作插入的結(jié)點(diǎn)作表頭結(jié)點(diǎn)951428220108615705142882NULL815702010head75121062106p02010p12010讓p0指向新結(jié)點(diǎn)即要插入的結(jié)點(diǎn),讓p1指向插入位置上的結(jié)點(diǎn)即表頭結(jié)點(diǎn)head=p0;p0->next=p1;2106插入的結(jié)點(diǎn)作表尾結(jié)點(diǎn)9693652951428220108615705142882NULL815702010head讓p0指向新結(jié)點(diǎn)即要插入的結(jié)點(diǎn),讓p1指向插入位置上的結(jié)點(diǎn)即表尾結(jié)點(diǎn)3652p01570p1NULL3652鏈表的插入結(jié)點(diǎn)操作p1->next=p0;p0->next=NULL;插入的結(jié)點(diǎn)作中間結(jié)點(diǎn)9262374951428220108615705142882NULL815702010head鏈表的插入結(jié)點(diǎn)操作讓p0指向新結(jié)點(diǎn)即要插入的結(jié)點(diǎn),讓p1指向插入位置上的結(jié)點(diǎn),p2指向p1的前驅(qū)結(jié)點(diǎn)2374p01570p11428p223741570p2->next=p0;p0->next=p1;鏈表的插入結(jié)點(diǎn)操作插入結(jié)點(diǎn)函數(shù)ST*insert(ST*head){ST*p0,*p1,*p2;p1=head;p0=(ST*)malloc(LEN);
scanf(“%d%d”,&p0->num,&p0->score);if(head==NULL){head=p0;p0->next=NULL;}else{while((p0->num>p1->num)&&(p1->next!=NULL)){p2=p1;p1=p1->next;}if(p0->num<=p1->num){if(head==p1)head=p0;elsep2->next=p0;p0->next=p1;}else{p1->next=p0;p0->next=NULL;}}n++;return(head);}//如果是空鏈表,則新結(jié)點(diǎn)是鏈表的表頭結(jié)點(diǎn)//找到要插入結(jié)點(diǎn)的位置//插入結(jié)點(diǎn)作表頭結(jié)點(diǎn)//插入結(jié)點(diǎn)作中間結(jié)點(diǎn)//插入結(jié)點(diǎn)作表尾結(jié)點(diǎn)//鏈表的結(jié)點(diǎn)個(gè)數(shù)加1鏈表的插入結(jié)點(diǎn)操作(1)插入鏈表的第一個(gè)結(jié)點(diǎn)ST*insert(ST*head){ST*p0,*p1,*p2;p1=head;p0=(ST*)malloc(LEN);
scanf(“%d%d”,&p0->num,&p0->score);
if(head==NULL){head=p0;p0->next=NULL;}
else{…}
n++;return(head);}headNULL1428p1p2p03861428NULL1428NULLn0
1鏈表的插入結(jié)點(diǎn)操作ST*insert(ST*head){ST*p0,*p1,*p2;p1=head;p0=(ST*)malloc(LEN);
scanf(“%d%d”,&p0->num,&p0->score);
if(head==NULL){…}
else{while((p0->num>p1->num)&&(p1->next!=NULL))
{…}
if(p0->num<=p1->num){if(head==p1)head=p0;
else…
p0->next=p1;}
else{…}}n++;return(head);}1428head1428386NULLn12p1p2p0
201020101428195(2)插入結(jié)點(diǎn)作表頭結(jié)點(diǎn)14282010鏈表的插入結(jié)點(diǎn)操作ST*insert(ST*head){ST*p0,*p1,*p2;p1=head;p0=(ST*)malloc(LEN);
scanf(“%d%d”,&p0->num,&p0->score);
if(head==NULL){…}else{while((p0->num>p1->num)&&(p1->next!=NULL)){p2=p1;p1=p1->next;}
if(p0->num<=p1->num){…}
else{p1->next=p0;p0->next=NULL;}}n++;return(head);}1570(3)插入結(jié)點(diǎn)作表尾結(jié)點(diǎn)20101428head20101951428386NULLp2p0n2p1582NULL201020101428315707.8類型定義符typedef的用法
當(dāng)用戶定義一種結(jié)構(gòu)體類型后,每次都需要用“struct
結(jié)構(gòu)體類型名”來(lái)定義相應(yīng)變量,稍顯麻煩。C語(yǔ)言不僅提供了豐富的數(shù)據(jù)類型,而且還允許由用戶自己定義類型說(shuō)明符,也就是說(shuō)允許由用戶為數(shù)據(jù)類型取“別名”。類型定義符ty
溫馨提示
- 1. 本站所有資源如無(wú)特殊說(shuō)明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁(yè)內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫(kù)網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。
最新文檔
- 2024年租賃合同租金調(diào)整及維修責(zé)任
- 2024版房地產(chǎn)買賣及租賃合同
- 如何做到合理膳食與健康
- 2024貸款分款協(xié)議書
- 2024年高端化妝品市場(chǎng)exclusivity合同
- 四川三河職業(yè)學(xué)院《馬克思主義經(jīng)濟(jì)學(xué)》2023-2024學(xué)年第一學(xué)期期末試卷
- 2024年高端智能設(shè)備采購(gòu)與技術(shù)轉(zhuǎn)讓合同
- 防水施工監(jiān)理旁站記錄
- 電氣培訓(xùn)資料:電纜敷設(shè)方式與代號(hào)
- 銅仁幼兒師范高等??茖W(xué)?!端帉W(xué)外語(yǔ)》2023-2024學(xué)年第一學(xué)期期末試卷
- 2024年中考語(yǔ)文名著閱讀《儒林外史》內(nèi)容簡(jiǎn)介、主要人物形象及相關(guān)練習(xí)
- 旅游商圈城市商圈oto系統(tǒng)
- 加速康復(fù)外科(ERAS)-骨科患者術(shù)后疼痛管理
- 研究生實(shí)驗(yàn)方案
- 三年級(jí)加法豎式題50道
- 阿里國(guó)際站合作協(xié)議
- 融資租賃物流方向行業(yè)報(bào)告
- 服務(wù)重點(diǎn)客戶
- 工業(yè)濾芯行業(yè)利潤(rùn)分析
- 2023年四川成都市初中學(xué)業(yè)水平考試生物試卷真題(答案詳解)
- 橋梁工程施工現(xiàn)場(chǎng)監(jiān)測(cè)方案
評(píng)論
0/150
提交評(píng)論