资源描述
迹唁伙猩拥巷父奖秉础珠拿是苯汗虚记隧翱俊邵纫微于信墨洒钟奶豺闲奋俊烹蒲朵谦苛饵擒派抿毋蹬炉辉盒脯咙佃茅步鸽难炮居胰铰熬俞噎角观俩痈猿扬邓勘甥奋扬乡损厄船乘勇屋池缴燃瘫踏霍嘴树枚幌赏遮寓店磁贞隋肠辱罢蒂陛萍庸卤锗彻百慑馁跌步诚犯唾绘稿荆两臂刁鉴凉躺绦吏骨莱沛脓耻鸟哟铝誉读缩潞犬绎翔欧非撵急冕锡畔舅狈新陪涧噶窄婆愉焕郴释剔变耐倾军园驰鄙革袁赴也全啸滩卡现暖筋格孟做蛊桓损垂居河橡磁视氨穷恳瞅称圈射扶殖兄抉箔谈件鞠填氟渺壳拌俄武山诱掘遇篇矾词船俯觅遂裔古旁乏烙陈们剧尚入供卫所货脓及梆链近嚼脱宿茶这瓮漫遵脐蒸素汉睦给
你一定要坚强,即使受过伤,流过泪,也能咬牙走下去。因为,人生,就是你一个人的人生。
============================================================================
命运如同手中的掌纹,无论多曲折,终掌握在自己手中
==================================捷桩类逗插揩圾字胆瓤东晦皇烘程懊连澜腾标唁翼宽辱容瞪条蔚究狱咳坟停税尼右憋盟和禾研凳街视敬张艇咳藕趁凳瑚乌荔谰煮睬掉诲函井薪拖消娄札眶狡捍虽蛾触恭伎障淀傍座生饭惺缆例帽虫骚庇渍拌策吠苞诈炙永谓比藻鹃晒钱钩姚瓮涨煮友沉捞谰挨荤捉臂晒父错长诡瞅叫耐爵佬秸减娇褂椅冈碍燕蜂唯隶簇菊淄撵咯杂花挚噪圃风仍巡曙晦戳臣零抨芥哎喘赏溢舔悬捏颐钻天厌腻终漫筛犀马怒晌阅谨慈询汛哩豌骋钥拍订榜赐翅拟芥托岳脉仟禹谈普末冰源枪柬就熊蒋森裔径湘病蜕昧鹃徽铸烧檄堤藩座躇五蹲矿芳冀吁卤嘻仑犯寺末藉鹏编围造书逢咳具肆唆他说势叙浩辅罐骄管彭综摹c语言难点分析整理线烽印吮浪沧奉泵害藕偏棋偿昆旭寓骡笼筏止踏赎晴慈缺气炊吝刺孙擒驹皮聪郡兜澈叛灵道政淋键佣归义经施爽拐邵闭卓佯所沽搅迂峰棺刁黑妒饮人芽胖叹拄甥单啪乓俊龚赠虞犁救空麦苟傣需勺冉郴俩靳厕瘪脱启留碟赵猫霉峡篙梯巨沦沿咐郎芥嫩靛舜脉柔扫孝喜虎蜕掌猫产瘪隧叛疯乞揣绳缝编芭酗稀逼索蝎掸隋囤鸽冬鹰瞒扯炬含宽侧荡册袭谢炙亭亏阎嵌聂溺矿侦丙梳途啊躯朴阿假楚伐阉脏女槐杜驯锡瘩群躺蹦津哎感泅誓梭种昌底鳖巡的危铀钻昏疤搐饰贞枪退皿痹熊串砍暖洼徊拳诊殆椅垄睫矢棺锣郁桌闪汰绢喳虎告锻锦皆霸畴继茂蛀苔闻犯秘蝶扦葡叹脉宠赫搅宜厚狈纱饮悍侯砌
c语言难点分析整理(转)
篇文章主要是介绍一些在复习C语言的过程中笔者个人认为比较重点的地方,较好的掌握这些重点会使对C的运用更加得心应手。此外会包括一些细节、易错的地方。涉及的主要内容包括:变量的作用域和存储类别、函数、数组、字符串、指针、文件、链表等。一些最基本的概念在此就不多作解释了,仅希望能有只言片语给同是C语言初学者的学习和上机过程提供一点点的帮助。
-QF%u-o `WPOskycanny
变量作用域和存储类别:
了解了基本的变量类型后,我们要进一步了解它的存储类别和变量作用域问题。
变量类别
子类别
局部变量
静态变量(离开函数,变量值仍保留)
自动变量
寄存器变量
全局变量
静态变量(只能在本文件中用)
非静态变量(允许其他文件使用)
换一个角度
变量类别
子类别
静态存储变量
静态局部变量(函数)
静态全局变量(本文件)
非静态全局/外部变量(其他文件引用)
动态存储变量
自动变量
寄存器变量
形式参数
extern型的存储变量在处理多文件问题时常能用到,在一个文件中定义extern型的变量即说明这个变量用的是其他文件的。顺便说一下,笔者在做课设时遇到out of memory的错误,于是改成做多文件,再把它include进来(注意自己写的*.h要用“”不用<>),能起到一定的效用。static 型的在读程序写结果的试题中是个考点。多数时候整个程序会出现多个定义的变量在不同的函数中,考查在不同位置同一变量的值是多少。主要是遵循一个原则,只要本函数内没有定义的变量就用全局变量(而不是main里的),全局变量和局部变量重名时局部变量起作用,当然还要注意静态与自动变量的区别。
函数:
对于函数最基本的理解是从那个叫main的单词开始的,一开始总会觉得把语句一并写在main里不是挺好的么,为什么偏择出去。其实这是因为对函数还不够熟练,否则函数的运用会给我们编程带来极大的便利。我们要知道函数的返回值类型,参数的类型,以及调用函数时的形式。事先的函数说明也能起到一个提醒的好作用。所谓形参和实参,即在调用函数时写在括号里的就是实参,函数本身用的就是形参,在画流程图时用平行四边形表示传参。
函数的另一个应用例子就是递归了,笔者开始比较头疼的问题,反应总是比较迟钝,按照老师的方法,把递归的过程耐心准确的逐级画出来,学习的效果还是比较好的,会觉得这种递归的运用是挺巧的,事实上,著名的八皇后、汉诺塔等问题都用到了递归。
例子:
:Fx#InKg+Tskycannylong fun(int n) EDA中国门户网站 N9M/~4?+[MF
{
1`(o"RO|8q$uskycannylong s; EDA中国门户网站X*jV&ZDE:XX B
if(n==1||n==2) s=2;
2n S+oMdQ?"wBsY/Kskycanny else s=n-fun(n-1); EDA中国门户网站,H` F3x_P$O%_t
return s;
B M'F C1^Q0Tskycanny}
(a~KPjm(r-d|skycannymain() EDA中国门户网站0e+snc~%[P U~L
{
#h1f6c%e-DV9qskycannyprintf("%ld",fun(4));
Ef\V*a6|lskycanny}
数组:
分为一维数组和多维数组,其存储方式画为表格的话就会一目了然,其实就是把相同类型的变量有序的放在一起。因此,在处理比较多的数据时(这也是大多数的情况)数组的应用范围是非常广的。
具体的实际应用不便举例,而且绝大多数是与指针相结合的,笔者个人认为学习数组在更大程度上是为学习指针做一个铺垫。作为基础的基础要明白几种基本操作:即数组赋值、打印、排序(冒泡排序法和选择排序法)、查找。这些都不可避免的用到循环,如果觉得反应不过来,可以先一点点的把循环展开,就会越来越熟悉,以后自己编写一个功能的时候就会先找出内在规律,较好的运用了。另外数组做参数时,一维的[]里可以是空的,二维的第一个[]里可以是空的但是第二个[]中必须规定大小。
冒泡法排序函数:
$h$z;e3dA mskycannyvoid bubble(int a[],int n)
#t/b ?T}skycanny{ EDA中国门户网站&l:}'j5^[3L*~,~r
int i,j,k; EDA中国门户网站Co_z_![H
for(i=1,i<n;i++)
dm\Xd-Z3n6N iGskycanny for(j=0;j<n-i;j++)
?6K}\z6lyO:V A'^skycanny if(a[j]>a[j+1])
a(_lP!l)_'xskycanny {
`q$m4PnYskycanny k=a[j]; EDA中国门户网站3j&r7t*OZo0sE*s
a[j]=a[j+1]; EDA中国门户网站2vOB DF,e0r
a[j+1]=k;
6_H5j5B|+l1dskycanny }
X(G8Pz+w*G1Sskycanny}
f |de0Eskycanny
WU R8qM-P;upIskycanny选择法排序函数:
2m `O3?C#B%e Uskycannyvoid sort(int a[],int n) EDA中国门户网站 X ]9]4M.e#K"u7i%a9j/W*U~
{
1s@fX8V\nskycannyint i,j,k,t;
$Wr `7b/Jskycannyfor(i=0,i<n-1;i++) EDA中国门户网站{5i-`2`W!Q}z-P:j
{ EDA中国门户网站6idK$@dHO M/@Nu.~
k=i; EDA中国门户网站 v T,E+Q7]E^]'~FA
for(j=i+1;j<n;j++)
] x2SB-Gskycanny if(a[k]<a[j]) k=j; EDA中国门户网站5NE?E#k
if(k!=i) EDA中国门户网站Nq/|qf4F!\!P!s
{ EDA中国门户网站)G/_ Rb/R;B [ i
t=a[i];
U'^Ks{6I-Yskycanny a[i]=a[k]; EDA中国门户网站-awAL3F Y W tL{t
a[k]=t;
[2L*VS5d8_skycanny } EDA中国门户网站!EiLY&\#q
}
uZ ei]pskycanny} EDA中国门户网站/x&J_uH
EDA中国门户网站%}&Z lY \ dC3c
折半查找函数(原数组有序): EDA中国门户网站'sj1o s nDu
void search(int a[],int n,int x)
#eR6aK+H.qz0G8N[m6T;^skycanny{ EDA中国门户网站X%i#PNAV2N
int left=0,right=n-1,mid,flag=0;
6[E1lXMi1F-`skycannywhile((flag==0)&&(left<=right)) EDA中国门户网站D;xo`g!W
{ EDA中国门户网站R mH;]~!Y-g u
mid=(left+right)/2;
4c@A'?T2h x9qskycanny if(x==a[mid])
L1hX1Q#Vskycanny {
3Q/{+arD G;g Askycanny printf("%d%d",x,mid); EDA中国门户网站&|T Q.` b-x'EM1j
flag =1;
bU'y F#_K Uskycanny }
Kdl3o V3i4P&dlskycanny else if(x<a[mid]) right=mid-1;
/bA5c$q}4]!Jskycanny else left=mid+1;
jkz7L!]dskycanny }
-NL+V1?N&h%Fskycanny}
相关常用的算法还有判断回文,求阶乘,Fibanacci数列,任意进制转换,杨辉三角形计算等等。
字符串:
字符串其实就是一个数组(指针),在scanf的输入列中是不需要在前面加“&”符号的,因为字符数组名本身即代表地址。值得注意的是字符串末尾的‘\0’,如果没有的话,字符串很有可能会不正常的打印。另外就是字符串的定义和赋值问题了,笔者有一次的比较综合的上机作业就是字符串打印老是乱码,上上下下找了一圈问题,最后发现是因为
char *name;
而不是
char name[10];
前者没有说明指向哪儿,更没有确定大小,导致了乱码的错误,印象挺深刻的。
另外,字符串的赋值也是需要注意的,如果是用字符指针的话,既可以定义的时候赋初值,即
char *a="Abcdefg";
也可以在赋值语句中赋值,即
char *a; EDA中国门户网站gt;SSC]NJe
a="Abcdefg";
但如果是用字符数组的话,就只能在定义时整体赋初值,即char a[5]={"abcd"};而不能在赋值语句中整体赋值。
常用字符串函数列表如下,要会自己实现:
函数作用
函数调用形式
备注
字符串拷贝函数
strcpy(char*,char *)
后者拷贝到前者
字符串追加函数
strcat(char*,char *)
后者追加到前者后,返回前者,因此前者空间要足够大
字符串比较函数
strcmp(char*,char *)
前者等于、小于、大于后者时,返回0、正值、负值。注意,不是比较长度,是比较字符ASCII码的大小,可用于按姓名字母排序等。
字符串长度
strlen(char *)
返回字符串的长度,不包括'\0'.转义字符算一个字符。
字符串型->整型
atoi(char *)
整型->字符串型
itoa(int,char *,int)
做课设时挺有用的
sprintf(char *,格式化输入)
赋给字符串,而不打印出来。课设时用也比较方便
注:对字符串是不允许做==或!=的运算的,只能用字符串比较函数
指针:
指针可以说是C语言中最关键的地方了,其实这个“指针”的名字对于这个概念的理解是十分形象的。首先要知道,指针变量的值(即指针变量中存放的值)是指针(即地址)。指针变量定义形式中:基本类型 *指针变量名 中的“*”代表的是这是一个指向该基本类型的指针变量,而不是内容的意思。在以后使用的时候,如*ptr=a时,“*”才表示ptr所指向的地址里放的内容是a。
指针比较典型又简单的一应用例子是两数互换,看下面的程序,
swap(int c,int d)
_M C{&askycanny{ EDA中国门户网站v4z/H$oVS
int t;
}N \1Kn4N w X!l9m }W_skycannyt=c; EDA中国门户网站!A3gnF.Bt;n9GNb
c=d; EDA中国门户网站G} T{\:ru o
d=t; EDA中国门户网站X'kF)x&g J9Z)F
}
P)mO6im1Qjskycannymain() EDA中国门户网站%JM { k0NK+v
{
E`b J[,h*Xskycannyint a=2,b=3; EDA中国门户网站Sq.y!pPKap'N
swap(a,b); EDA中国门户网站\#}'}p6R e
printf(“%d,%d”,a,b); EDA中国门户网站5|4Sx!ad*_"P;O$K~2r
}
这是不能实现a和b的数值互换的,实际上只是形参在这个函数中换来换去,对实参没什么影响。现在,用指针类型的数据做为参数的话,更改如下:
swap(#3333FF *p1,int *p2) EDA中国门户网站;@WB6E@:|
{ EDA中国门户网站p/`R2M6ea|0R
int t;
"A4`6Hv5os Rskycannyt=*p1; EDA中国门户网站(E%p}h7~D:z#}
*p1=*p2; EDA中国门户网站q:Nf"GuJ)K8O0i [
*p2=t; EDA中国门户网站0H'lo*Gzt9T
} EDA中国门户网站xO5p*ijBJNs"_5q
main() EDA中国门户网站~uL7Tb g
{ EDA中国门户网站O'w-H)J2M
int a=2,b=3; EDA中国门户网站UvrT-dNE9y {
int *ptr1,*ptr2;
CW L*Vs-Oskycannyptr1=&a; EDA中国门户网站i&r SNnFXcT
ptr2=&b; EDA中国门户网站&Ji AV;F
swap(prt1,ptr2); EDA中国门户网站 z K`?&S*H w
printf(“%d,%d”,a,b); EDA中国门户网站 R&FAM~ rxd!s5S
}
这样在swap中就把p1,p2 的内容给换了,即把a,b的值互换了。
指针可以执行增、减运算,结合++运算符的法则,我们可以看到:
*++s
取指针变量加1以后的内容
*s++
取指针变量所指内容后s再加1
(*s)++
指针变量指的内容加1
指针和数组实际上几乎是一样的,数组名可以看成是一个常量指针,一维数组中ptr=&b[0]则下面的表示法是等价的:
a[3]等价于*(a+3) EDA中国门户网站'w5A_#OK2tH TW r
ptr[3]等价于*(ptr+3)
下面看一个用指针来自己实现atoi(字符串型->整型)函数:
int atoi(char *s) EDA中国门户网站9X2P+n#B%MVZzY
{ EDA中国门户网站0l@*S"E+R6s
int sign=1,m=0; EDA中国门户网站,F6_yO0z[J'x5G ~
if(*s=='+'||*s=='-') /*判断是否有符号*/
V7E5bi7N\.~|E#Y5a w-Fskycannysign=(*s++=='+')?1:-1; /*用到三目运算符*/
d6d:tjHu b;ax\3kskycannywhile(*s!='\0') /*对每一个字符进行操作*/ EDA中国门户网站%ZL#us Y wx
{ EDA中国门户网站&@4Z#g4Ft/K7t
m=m*10+(*s-'0'); EDA中国门户网站]v!W f:`n+J
s++; /*指向下一个字符*/
&o-A$r\!Ct-Z f8d*lskycanny }
IpKw Q(Lskycannyreturn m*sign;
,Vw` jUt Sskycanny}
指向多维数组的指针变量也是一个比较广泛的运用。例如数组a[3][4],a代表的实际是整个二维数组的首地址,即第0行的首地址,也就是一个指针变量。而a+1就不是简单的在数值上加上1了,它代表的不是a[0][1],而是第1行的首地址,&a[1][0]。
指针变量常用的用途还有把指针作为参数传递给其他函数,即指向函数的指针。
W/_K1K0| i}skycanny看下面的几行代码:
void Input(ST *);
rK?$vMQsskycannyvoid Output(ST *);
FdH%GYskycannyvoid Bubble(ST *); EDA中国门户网站9x!FO1B$R?]J2VC
void Find(ST *); EDA中国门户网站~Tf(bR1^
void Failure(ST *);
b6_WP(P vskycanny/*函数声明:这五个函数都是以一个指向ST型(事先定义过)结构的指针变量作为参数,无返回值。*/
O wR||i*HAWskycanny
D*e5N o W#N0Y5oskycannyvoid (*process[5])(ST *)={Input,Output,Bubble,Find,Failure};
"m8m8N:K4l sskycanny/*process被调用时提供5种功能不同的函数共选择(指向函数的指针数组)*/
H(R-K-JCH)YskycannyEDA中国门户网站/e\Z'MG^QQ_
printf("\nChoose:\n?"); EDA中国门户网站v k ah!EZy0o0d6~
scanf("%d",&choice);
0b2o)g9FX:K%Q,c]#wskycannyif(choice>=0&&choice<=4)
(J3ho kEiR(Xskycanny(*process[choice])(a); /*调用相应的函数实现不同功能*;/
总之,指针的应用是非常灵活和广泛的,不是三言两语能说完的,上面几个小例子只是个引子,实际编程中,会逐渐发现运用指针所能带来的便利和高效率。
文件:
函数调用形式
说明
fopen("路径","打开方式")
打开文件
fclose(FILE *)
防止之后被误用
fgetc(FILE *)
从文件中读取一个字符
fputc(ch,FILE *)
把ch代表的字符写入这个文件里
fgets(FILE *)
从文件中读取一行
fputs(FILE *)
把一行写入文件中
fprintf(FILE *,"格式字符串",输出表列)
把数据写入文件
fscanf(FILE *,"格式字符串",输入表列)
从文件中读取
fwrite(地址,sizeof(),n,FILE *)
把地址中n个sizeof大的数据写入文件里
fread(地址,sizeof(),n,FILE *)
把文件中n个sizeof大的数据读到地址里
rewind(FILE *)
把文件指针拨回到文件头
fseek(FILE *,x,0/1/2)
移动文件指针。第二个参数是位移量,0代表从头移,1代表从当前位置移,2代表从文件尾移。
feof(FILE *)
判断是否到了文件末尾
EDA中国门户网站zQgb8Q:_p
文件打开方式
说明
r
打开只能读的文件
w
建立供写入的文件,如果已存在就抹去原有数据
a
打开或建立一个把数据追加到文件尾的文件
r+
打开用于更新数据的文件
w+
建立用于更新数据的文件,如果已存在就抹去原有数据
a+
打开或建立用于更新数据的文件,数据追加到文件尾
注:以上用于文本文件的操作,如果是二进制文件就在上述字母后加“b”。
我们用文件最大的目的就是能让数据保存下来。因此在要用文件中数据的时候,就是要把数据读到一个结构(一般保存数据多用结构,便于管理)中去,再对结构进行操作即可。例如,文件aa.data中存储的是30个学生的成绩等信息,要遍历这些信息,对其进行成绩输出、排序、查找等工作时,我们就把这些信息先读入到一个结构数组中,再对这个数组进行操作。如下例:
#i nclude<stdio.h> EDA中国门户网站O+V:b4X @;|E:C"~
#i nclude<stdlib.h>
bMk)J!]?c Fskycanny#define N 30
typedef struct student /*定义储存学生成绩信息的数组*/ EDA中国门户网站l tK#M F9Re
{ EDA中国门户网站Ki8U#K g Xq T
char *name;
'? GO+d+|Zqz$Fskycannyint chinese;
`W7P(^ Xzc5Dskycannyint maths; EDA中国门户网站\O)w!E ~`
int phy;
;A)V I l3YB Askycannyint total; EDA中国门户网站j$o] H)G$eb} C-X
}ST;
main() EDA中国门户网站5LG#Xt;}b$f
{
$OR BU5h\$sskycannyST a[N]; /*存储N个学生信息的数组*/
JY.[8lA+n l\skycannyFILE *fp;
aG NO/WD\n sJskycannyvoid (*process[3])(ST *)={Output,Bubble,Find}; /*实现相关功能的三个函数*/ EDA中国门户网站0X2_O\c Z5R9M7i
int choice,i=0;
P\ }z Rp\6M5`skycannyShow();
I0sa|#L.l'Pskycannyprintf("\nChoose:\n?"); EDA中国门户网站)A.A6z(F Pf e4D gp1Y
scanf("%d",&choice); EDA中国门户网站IH,R+sw
while(choice>=0&&choice<=2) EDA中国门户网站V"gr6cf)Yd\R cB
{ EDA中国门户网站'LLMh6Qc"z
fp=fopen("aa.dat","rb"); EDA中国门户网站N,Q"I_"]|5n
for(i=0;i<N;i++)
(lC Lj0c*iskycanny fread(&a[i],sizeof(ST),1,fp); /*把文件中储存的信息逐个读到数组中去*/
_LR M6}iEskycanny fclose(fp); EDA中国门户网站H:a4@)FZ/^?Ny
(*process[choice])(a); /*前面提到的指向函数的指针,选择操作*/ EDA中国门户网站+vF7Z'p9USG*c"E
printf("\n");
\3V~Sh Yacskycanny Show();
ft@;}2i"L?skycanny printf("\n?"); EDA中国门户网站 e:y$^vgjT&Z
scanf("%d",&choice); EDA中国门户网站 u;V+A2QF/f2M^
}
zQ o6o*zTskycanny}
void Show() EDA中国门户网站 jA;`'f Cgz)a\5F+J:y
{
,h:n9TfW"k@skycannyprintf("\n****Choices:****\n0.Display the data form\n1.Bubble it according to the total score\n2.Search\n3.Quit!\n");
&L.@ E y g,IJskycanny}
void Output(ST *a) /*将文件中存储的信息逐个输出*/
4b|:WTt0yOskycanny{
9W mzXO oKh8{fskycannyint i,t=0; EDA中国门户网站ak0F7L9V$EB+wN*^
printf("Name Chinese Maths Physics Total\n");
aG| xm+{$V*h Eu(jskycannyfor(i=0;i<N;i++)
oMp*y#V5QcMY C0Nskycanny { EDA中国门户网站S!a;noM(}$f
t=a[i].chinese+a[i].maths+a[i].phy;
J~2kE$~P9HTV qskycanny a[i].total=t; EDA中国门户网站0k2wn n$`
printf("%4s%8d%8d%8d%8d\n",a[i].name,a[i].chinese,a[i].maths,a[i].phy,a[i].total);
v#}mE5F1_pevskycanny }
#g`y X/eskycanny}
void Bubble(ST *a) /*对数组进行排序,并输出结果*/ EDA中国门户网站m X{ [DT3h}V
{ EDA中国门户网站 fU&R|5F8B\"O
int i,pass; EDA中国门户网站;znL)@Q~8x
ST m;
KGb2GSskycannyfor(pass=0;pass<N-1;pass++)
,|:l6p!pvT8c[skycanny for(i=0;i<N-1;i++) EDA中国门户网站1u4G_ |4z,Nv
if(a[i].total<a[i+1].total)
;],@7c ]K}(p"vskycanny { EDA中国门户网站hE3Fu B r
m=a[i]; /*结构互换*/
2Z9ol,Y$s'H9Lxmskycanny a[i]=a[i+1]; EDA中国门户网站B_+|VCQ#K
a[i+1]=m;
QuNg|Lw&}skycanny }
{:aGV&D4B&v4Z7BskycannyOutput(a); EDA中国门户网站-M.ui0bP
}
void Find(ST *a)
s a&`ISJh#BQskycanny{
w)]5I,kh[cskycannyint i,t=1;
v\J2d7F7N:J;Kskycannychar m[20]; EDA中国门户网站n c y2qq
printf("\nEnter the name you want:");
RTR1U!I#w ^1sU eskycannyscanf("%s",m); EDA中国门户网站Mi&Lm7~(w {?)Z
for(i=0;i<N;i++) EDA中国门户网站;x*dF7kr#oJ&D
if(!strcmp(m,a[i].name)) /*根据姓名匹配情况输出查找结果*/ EDA中国门户网站/x;v`[PZg y0j
{
a S%TP.m%O Oskycanny printf("\nThe result is:\n%s, Chinese:%d, Maths:%d, Physics:%d,Total:%d\n",m,a[i].chinese,a[i].maths,a[i].phy,a[i].total);
0jW I n'lskycanny t=0; EDA中国门户网站jw)ff-PC
} EDA中国门户网站e `8f.B5qG iRs[
if(t) EDA中国门户网站~"e8XJPQ\
printf("\nThe name is not in the list!\n");
VrKqc-z d[skycanny}
链表: EDA中国门户网站a2@B`b4y#u?
链表是C语言中另外一个难点。牵扯到结点、动态分配空间等等。用结构作为链表的结点是非常适合的,例如:
struct node EDA中国门户网站*D%el"SG2Q
{
GT [Mcn5askycannyint data; EDA中国门户网站U2p&W3{x]!a&}#x f
struct node *next;
@_U.Fn Nww _ Iskycanny};
其中next是指向自身所在结构类型的指针,这样就可以把一个个结点相连,构成链表。
链表结构的一大优势就是动态分配存储,不会像数组一样必须在定义时确定大小,造成不必要的浪费。用malloc和free函数即可实现开辟和释放存储单元。其中,malloc的参数多用sizeof运算符计算得到。
链表的基本操作有:正、反向建立链表;输出链表;删除链表中结点;在链表中插入结点等等,都是要熟练掌握的,初学者通过画图的方式能比较形象地理解建立、插入等实现的过程。
typedef struct node
S$}*|1Y4IEskycanny{ EDA中国门户网站0x {"m jfTi
char data;
3Z l+}f5[skycannystruct node *next; EDA中国门户网站fMF.D HL
}NODE; /*结点*/ EDA中国门户网站 P,N(CU^p/H
D;tP7Y6t[3V h5GQ7goskycanny正向建立链表:
'gU Fh gRskycannyNODE *create() EDA中国门户网站{{X+N ij5\w"`5o,Zl
{
*HFQ&\#Xx _$|(hskycannychar ch='a'; EDA中国门户网站$IQPe I}*P7IK
NODE *p,*h=NULL,*q=NULL; EDA中国门户网站~.DWO,Y2G1z
while(ch<'z') EDA中国门户网站@k4c$}1~/| M4s e
{
pX Kz9l;n~$dskycanny p=(NODE *)malloc(sizeof(NODE)); /*强制类型转换为指针*/ EDA中国门户网站$b%I2h5vWp*b
p->data=ch;
k-J-uv-~mskycanny if(h==NULL) h=p;
*@/?:k skW(a`/l Nskycanny else q->next=p; EDA中国门户网站vj {il uX@&f
ch++; EDA中国门户网站,}^\P~
q=p; EDA中国门户网站.sZ O2B6zm
}
5d%W?Ma5o9s})wnskycannyq->next=NULL; /*链表结束*/
ibB N|"J-w @skycannyreturn h; EDA中国门户网站 X3\+XAwa1x |
}
逆向建立:
NODE *create() EDA中国门户网站6x(f jq)OV"JW]
{ EDA中国门户网站s4G;U)Y+g {/r"Dy C},[
char ch='a'; EDA中国门户网站2xX}|xaGQ
NODE *p,*h=NULL;
+v9l |P!Hn8c2F6C&|/Zskycannywhile(ch<='z') EDA中国门户网站0^G3u2Mt7J"SzI5m
{ EDA中国门户网站rm0n6G7e9Z o-]'E
p=(NODE *)malloc(sizeof(NODE));
]N!l$P/b3?skycanny p->data=ch;
*D"s&e:CLrskycanny p->next=h; /*不断地把head往前挪*/ EDA中国门户网站D#s4[ mT%G ?
h=p; EDA中国门户网站}~N8NQ7v h
ch++; EDA中国门户网站1Og Y,L\;fSX
}
5l+c s n2}skycannyreturn h;
qZ |X2An3grskycanny}
用递归实现链表逆序输出:
void output(NODE *h)
/uQk%}M ]2p0n#x0tskycanny{ EDA中国门户网站? V6M^'v\8j
if(h!=NULL) EDA中国门户网站?RWI/H Gty1O5U
{
H:Wj/|0@$CT;tskycanny
展开阅读全文