让心去旅行!

C语言难点分析整理 转

上一篇 / 下一篇  2006-10-07 16:08:11 / 天气: 晴朗 / 心情: 高兴

C语言难点分析整理
 
EDA中国门户网站"t {p7QV`a7t0^B#v
这篇文章主要是介绍一些在复习C语言的过程中笔者个人认为比较重点的地方,较好的掌握这些重点会使对C的运用更加得心应手。此外会包括一些细节、易错的地方。涉及的主要内容包括:变量的作用域和存储类别、函数、数组、字符串、指针、文件、链表等。一些最基本的概念在此就不多作解释了,仅希望能有只言片语给同是C语言初学者的学习和上机过程提供一点点的帮助。
5B1\T5]i:u.D/U0

变量作用域和存储类别:

了解了基本的变量类型后,我们要进一步了解它的存储类别和变量作用域问题。

变量类别子类别
局部变量静态变量(离开函数,变量值仍保留)
自动变量
寄存器变量
全局变量静态变量(只能在本文件中用)
非静态变量(允许其他文件使用)

换一个角度

变量类别子类别
静态存储变量静态局部变量(函数)
静态全局变量(本文件)
非静态全局/外部变量(其他文件引用)
动态存储变量自动变量
寄存器变量
形式参数

extern型的存储变量在处理多文件问题时常能用到,在一个文件中定义extern型的变量即说明这个变量用的是其他文件的。顺便说一下,笔者在做课设时遇到out of memory的错误,于是改成做多文件,再把它include进来(注意自己写的*.h要用“”不用<>),能起到一定的效用。static 型的在读程序写结果的试题中是个考点。多数时候整个程序会出现多个定义的变量在不同的函数中,考查在不同位置同一变量的值是多少。主要是遵循一个原则,只要本函数内没有定义的变量就用全局变量(而不是main里的),全局变量和局部变量重名时局部变量起作用,当然还要注意静态与自动变量的区别。

函数:

对于函数最基本的理解是从那个叫main的单词开始的,一开始总会觉得把语句一并写在main里不是挺好的么,为什么偏择出去。其实这是因为对函数还不够熟练,否则函数的运用会给我们编程带来极大的便利。我们要知道函数的返回值类型,参数的类型,以及调用函数时的形式。事先的函数说明也能起到一个提醒的好作用。所谓形参和实参,即在调用函数时写在括号里的就是实参,函数本身用的就是形参,在画流程图时用平行四边形表示传参。

函数的另一个应用例子就是递归了,笔者开始比较头疼的问题,反应总是比较迟钝,按照老师的方法,把递归的过程耐心准确的逐级画出来,学习的效果还是比较好的,会觉得这种递归的运用是挺巧的,事实上,著名的八皇后、汉诺塔等问题都用到了递归。

例子:
R4~E0~`#XTt0long fun(int n)
#P%HHe;a'|QD e1O0{EDA中国门户网站m.F2D%bz6]G$f
long s;EDA中国门户网站|]$c&`;D
if(n==1||n==2) s=2;
;R?(U#r&K.I/hj&s0   else s=n-fun(n-1);EDA中国门户网站*H|"_;Gq!E'?
return s;EDA中国门户网站3u6K#xg(Ef:d?"G
}EDA中国门户网站4VYW.L:z:w~/[
main()EDA中国门户网站1w'b`/y x,Z0Bv
{EDA中国门户网站 S:F/z4p@ e W
printf("%ld",fun(4));
Z N-bc:P F x0}

 

数组:

分为一维数组和多维数组,其存储方式画为表格的话就会一目了然,其实就是把相同类型的变量有序的放在一起。因此,在处理比较多的数据时(这也是大多数的情况)数组的应用范围是非常广的。

具体的实际应用不便举例,而且绝大多数是与指针相结合的,笔者个人认为学习数组在更大程度上是为学习指针做一个铺垫。作为基础的基础要明白几种基本操作:即数组赋值、打印、排序(冒泡排序法和选择排序法)、查找。这些都不可避免的用到循环,如果觉得反应不过来,可以先一点点的把循环展开,就会越来越熟悉,以后自己编写一个功能的时候就会先找出内在规律,较好的运用了。另外数组做参数时,一维的[]里可以是空的,二维的第一个[]里可以是空的但是第二个[]中必须规定大小。

冒泡法排序函数:EDA中国门户网站 r.`;Re)}.[7cBn
void bubble(int a[],int n)
"L3G m e(`]?2v:t\s*l0{EDA中国门户网站l,F0} |isz
int i,j,k;EDA中国门户网站x:j A-m{2uk5Yf
for(i=1,i<n;i++)EDA中国门户网站 _6^4P FVT,m
   for(j=0;j<n-i;j++)EDA中国门户网站.n:q?9U~l.Cd v
   if(a[j]>a[j+1])
!B^!pq5g0    {EDA中国门户网站zQ_ |{$ypK#v
    k=a[j];
{4_)_7{.s5W;T!w(h0       a[j]=a[j+1];
.?%bRc|"L:],U0       a[j+1]=k;
/Ks'Bh4S0       }EDA中国门户网站O4iM(@{{XZ
}EDA中国门户网站hbRbr1d3W&V{Yf
EDA中国门户网站%MVXHj
选择法排序函数:
_1[ `1uH cEg0void sort(int a[],int n)EDA中国门户网站BJFA]Z!X.vr4|)d
{
b/x6p yCp4l l0int i,j,k,t;
^H"G)}+D;` T0for(i=0,i<n-1;i++)EDA中国门户网站x1M[hJ|3a
   {EDA中国门户网站#kLdQV5g"J4s
   k=i;
Y&ht DcE!u&w"G0   for(j=i+1;j<n;j++)EDA中国门户网站d7|{3S.c
      if(a[k]<a[j]) k=j;
!l/t A w+ztp0      if(k!=i)
0Z lBE&W0         {EDA中国门户网站E!\R_gG
         t=a[i];
+Urc h+GuA+~KA0         a[i]=a[k];EDA中国门户网站.Zc tyBl(mQ.G-S{
         a[k]=t;
Wd~!ckD5vS tn0         }EDA中国门户网站 c)xR#s2O
   }
H-[7|V1s*ylt.L0}
EX&r$S&E}0EDA中国门户网站J,U2y*R~(x*^
折半查找函数(原数组有序):EDA中国门户网站y0|I?A}g
void search(int a[],int n,int x)
@s T)X j[0{EDA中国门户网站H haaH
int left=0,right=n-1,mid,flag=0;
2r p7I%Z8GI9[:a0while((flag==0)&&(left<=right))
U dNs:z z#y R0   {EDA中国门户网站&T@0rcj na6g&yV
   mid=(left+right)/2;EDA中国门户网站4qj4^*EF#m S:Y
   if(x==a[mid])EDA中国门户网站}+jZ l5sQ RSVu)M
      {EDA中国门户网站h"cL5bAd ]
      printf("%d%d",x,mid);
%h0Z Xa`xy;o(qD0      flag =1;EDA中国门户网站XQ3| z^W KpX
      }
#o8IV4c8b5x5k`0      else if(x<a[mid]) right=mid-1;EDA中国门户网站g}ueX'c TIh:Wa
                   else left=mid+1;
1@:Mcb7}C0   }EDA中国门户网站RsP/^D
}

相关常用的算法还有判断回文,求阶乘,Fibanacci数列,任意进制转换,杨辉三角形计算等等

字符串:

字符串其实就是一个数组(指针),在scanf的输入列中是不需要在前面加“&”符号的,因为字符数组名本身即代表地址。值得注意的是字符串末尾的‘\0’,如果没有的话,字符串很有可能会不正常的打印。另外就是字符串的定义和赋值问题了,笔者有一次的比较综合的上机作业就是字符串打印老是乱码,上上下下找了一圈问题,最后发现是因为

char *name;

而不是

char name[10];

前者没有说明指向哪儿,更没有确定大小,导致了乱码的错误,印象挺深刻的。

另外,字符串的赋值也是需要注意的,如果是用字符指针的话,既可以定义的时候赋初值,即

char *a="Abcdefg";

也可以在赋值语句中赋值,即

char *a;
|,k%_ DX(n d0a="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)EDA中国门户网站1c@r^e:n"b
{EDA中国门户网站OT.x7J9Jj
int t;EDA中国门户网站(pl'JLer m0iPgl
t=c;EDA中国门户网站,_%g xS)KH9N
c=d;EDA中国门户网站:V+E$K']ak
d=t;
P[*y&U'V.Y0}
fI_&]D/_"nY0main()
G&fM7nl,} n s0{
M!b!}[I0int a=2,b=3;EDA中国门户网站qJvJ'F
swap(a,b);
%kA _!T$ElSO aL0printf(“%d,%d”,a,b);EDA中国门户网站]'x A1D&L:_3u+I zE
}

这是不能实现a和b的数值互换的,实际上只是形参在这个函数中换来换去,对实参没什么影响。现在,用指针类型的数据做为参数的话,更改如下:

swap(#3333FF *p1,int *p2)EDA中国门户网站?fVw#c6DS}
{
%g*g7l:y"BYioSBf0int t;EDA中国门户网站oo&c$uF ?I@J8u/f
t=*p1;
z Y D~u0*p1=*p2;
-s5EqD3BZz0*p2=t;EDA中国门户网站krU'sK Fh6Mr!N
}
.w2~&a$z _wj0main()EDA中国门户网站wE e!fPVw4p
{EDA中国门户网站6n-J-L A|0J ZU'x
int a=2,b=3;EDA中国门户网站 t3G4c4wv\"I?0R
int *ptr1,*ptr2;EDA中国门户网站1K!r7D4`C6j
ptr1=&a;
}]7[%B0U0ptr2=&b;EDA中国门户网站!f0b%Mt{4n,ra
swap(prt1,ptr2);EDA中国门户网站h^\b e6|%J
printf(“%d,%d”,a,b);
5N3riv F/q*[0}

这样在swap中就把p1,p2 的内容给换了,即把a,b的值互换了。

指针可以执行增、减运算,结合++运算符的法则,我们可以看到:

*++s

取指针变量加1以后的内容

*s++取指针变量所指内容后s再加1
(*s)++指针变量指的内容加1

指针和数组实际上几乎是一样的,数组名可以看成是一个常量指针,一维数组中ptr=&b[0]则下面的表示法是等价的:

a[3]等价于*(a+3)EDA中国门户网站^x z%Np
ptr[3]等价于*(ptr+3)

下面看一个用指针来自己实现atoi(字符串型->整型)函数:

int atoi(char *s)
K H9Vf}j8p8Q0{
Mbxl`*Gc:^0int sign=1,m=0;EDA中国门户网站 E0q1yY u!U,`"Hn
if(*s=='+'||*s=='-') /*判断是否有符号*/EDA中国门户网站3X0LfyW&dj [
sign=(*s++=='+')?1:-1; /*用到三目运算符*/
$uT(gEi)W'v0while(*s!='\0') /*对每一个字符进行操作*/
s abB?0   {EDA中国门户网站CtY6Sr
   m=m*10+(*s-'0');EDA中国门户网站V1_UeW5d]
   s++; /*指向下一个字符*/EDA中国门户网站;w t:l:Nv4`
   }
fW0?|v'T#\,B0return m*sign;EDA中国门户网站+bRi3HLe+L
}

指向多维数组的指针变量也是一个比较广泛的运用。例如数组a[3][4],a代表的实际是整个二维数组的首地址,即第0行的首地址,也就是一个指针变量。而a+1就不是简单的在数值上加上1了,它代表的不是a[0][1],而是第1行的首地址,&a[1][0]。

指针变量常用的用途还有把指针作为参数传递给其他函数,即指向函数的指针EDA中国门户网站 F Z/i.jU'@
看下面的几行代码:

void Input(ST *);EDA中国门户网站A)e,R'u Fs.]
void Output(ST *);EDA中国门户网站G?9KZnhh2No
void Bubble(ST *);EDA中国门户网站"V9Kv$cb"ur5z
void Find(ST *);EDA中国门户网站+Y2`YH|(P!j['Q*P+k
void Failure(ST *);
!d z*l yZ0/*函数声明:这五个函数都是以一个指向ST型(事先定义过)结构的指针变量作为参数,无返回值。*/
2ZLvc ^:L5k,Wv R0
!_q!B?4T l0void (*process[5])(ST *)={Input,Output,Bubble,Find,Failure};EDA中国门户网站3A+A9t%I/dnUR o C
/*process被调用时提供5种功能不同的函数共选择(指向函数的指针数组)*/EDA中国门户网站B)g6V$R/h-?)o)tR H*j

m+WtU T3A[;[K]0printf("\nChoose:\n?");
!_3u8vh9PY&Ra0D C n0h0scanf("%d",&choice);EDA中国门户网站h@-~b t#e(R%m
if(choice>=0&&choice<=4)EDA中国门户网站e A)oi$tG$e'B
(*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 *)判断是否到了文件末尾

D#C?.T6Z\;h0
文件打开方式说明
r打开只能读的文件
w建立供写入的文件,如果已存在就抹去原有数据
a打开或建立一个把数据追加到文件尾的文件
r+打开用于更新数据的文件
w+建立用于更新数据的文件,如果已存在就抹去原有数据
a+打开或建立用于更新数据的文件,数据追加到文件尾

注:以上用于文本文件的操作,如果是二进制文件就在上述字母后加“b”。

我们用文件最大的目的就是能让数据保存下来。因此在要用文件中数据的时候,就是要把数据读到一个结构(一般保存数据多用结构,便于管理)中去,再对结构进行操作即可。例如,文件aa.data中存储的是30个学生的成绩等信息,要遍历这些信息,对其进行成绩输出、排序、查找等工作时,我们就把这些信息先读入到一个结构数组中,再对这个数组进行操作。如下例:

#include<stdio.h>EDA中国门户网站 J1z1f-{,o~:N V(J)E8W
#include<stdlib.h>
5H6R cOX;t5H_/NO0#define N 30

typedef struct student /*定义储存学生成绩信息的数组*/EDA中国门户网站+sH1Qh.c c5U8Uc M
{EDA中国门户网站(_~+}(FET
char *name;
p%hT9I)B;`!x6z0int chinese;
D;N@]j(n|/Y;D0int maths;EDA中国门户网站;z'Z A8yz2Z
int phy;EDA中国门户网站.Ugt)V4SO/_
int total;EDA中国门户网站)L@Q5`5CO D
}ST;

main()EDA中国门户网站Qo@%`0V1P1pzWC
{
D/W2rf5Gp.I$~y t(V0ST a[N]; /*存储N个学生信息的数组*/
7|&T deK:L0FILE *fp;
@@~.V%{0void (*process[3])(ST *)={Output,Bubble,Find}; /*实现相关功能的三个函数*/
i!y-m+~+]9|A;R`0int choice,i=0;
p"`Z{\ H0Show();
.E$^#u Wl \ \VE0uK0printf("\nChoose:\n?");
)N!?0V_jUR0scanf("%d",&choice);EDA中国门户网站$fC,G6R ^
while(choice>=0&&choice<=2)EDA中国门户网站G5se{$LK
   {EDA中国门户网站!D*kl/U"r;Z^'?N
   fp=fopen("aa.dat","rb");
U'~wMn0   for(i=0;i<N;i++)EDA中国门户网站*|)CH;z Xv3uyKT
      fread(&a[i],sizeof(ST),1,fp); /*把文件中储存的信息逐个读到数组中去*/EDA中国门户网站^A#f2\9Ri7eN
   fclose(fp);EDA中国门户网站K;r%W&?K{b-A
   (*process[choice])(a); /*前面提到的指向函数的指针,选择操作*/EDA中国门户网站3t1q X,L I*B"KH4L z;n
   printf("\n");
g,nS C-Zt4n%k0   Show();EDA中国门户网站5ucz+Ll5Y
   printf("\n?");EDA中国门户网站 y"~S&D0b8]
   scanf("%d",&choice);EDA中国门户网站g*v7po)Xtm
   }EDA中国门户网站kW%? SK;|&RErx
}

void Show()EDA中国门户网站 g6r-bEn:Ic;Q
{EDA中国门户网站o s?U9k#xt/TC pu
printf("\n****Choices:****\n0.Display the data form\n1.Bubble it according to the total score\n2.Search\n3.Quit!\n");
Fm y|ub6Ku0}

void Output(ST *a) /*将文件中存储的信息逐个输出*/
-T4f;a*K5h m0{EDA中国门户网站nl^:f/KF
int i,t=0;
%hp{.O7P["Z1T0printf("Name Chinese Maths Physics Total\n");
:SlR J;^^h0for(i=0;i<N;i++)EDA中国门户网站;F_Xs0Lg2t
   {EDA中国门户网站kgd;O5B?tl S
   t=a[i].chinese+a[i].maths+a[i].phy;EDA中国门户网站p!R:W#@n E4\ [vb
   a[i].total=t;EDA中国门户网站z0u-Y/jU!o }7S4]
   printf("%4s%8d%8d%8d%8d\n",a[i].name,a[i].chinese,a[i].maths,a[i].phy,a[i].total);EDA中国门户网站!bOq;|e Jfs
   }EDA中国门户网站6JRM7K_(W W&s
}

void Bubble(ST *a) /*对数组进行排序,并输出结果*/
p5CusP(j0{EDA中国门户网站8c-k9J It
int i,pass;
)Uh5|^ @w*r0ST m;
'dV(Q T/MNYR.d-XJ0for(pass=0;pass<N-1;pass++)
/dFH%jS3Bll.L0   for(i=0;i<N-1;i++)
C.`(|(jHgan2n0      if(a[i].total<a[i+1].total)EDA中国门户网站ky#^z5Q)m.]
         {EDA中国门户网站V[0Z'f7dU
         m=a[i]; /*结构互换*/EDA中国门户网站)DvN+N/e2k
         a[i]=a[i+1];EDA中国门户网站7lDW0ZF W:[Q'| K
         a[i+1]=m;EDA中国门户网站 N jFkY;R W0?s
         }
]ddTSK}&}W}0Output(a);
'QM*al-Q{0}

void Find(ST *a)EDA中国门户网站qF QrN:ALU0\
{
-{5f'A^6ng0int i,t=1;
mN|yE'\(N'G0L0char m[20];
#eD!P co}D"m0printf("\nEnter the name you want:");EDA中国门户网站Y8wo:W!BbSF&?
scanf("%s",m);
A1H!q+Rk!~H {|O0for(i=0;i<N;i++)
|4BT,{cV w"tG0   if(!strcmp(m,a[i].name)) /*根据姓名匹配情况输出查找结果*/EDA中国门户网站 X)ZU9g,tg{'j
   {
Qn7F1\hI0   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);EDA中国门户网站4KC$x)c6{$M
   t=0;
} I:q8t2y0   }EDA中国门户网站'm mL1A M%K"v
if(t)EDA中国门户网站C&i)X#df d-m
   printf("\nThe name is not in the list!\n");
vc0?r*}n ~4q`)M:i0}

链表:
Fz VlhjyX0链表是C语言中另外一个难点。牵扯到结点、动态分配空间等等。用结构作为链表的结点是非常适合的,例如:

struct nodeEDA中国门户网站^B{$bF
{
cksAKE r$[Q0int data;
n"~8p"d?;g4U0struct node *next;EDA中国门户网站!}v]y*?8e
};

其中next是指向自身所在结构类型的指针,这样就可以把一个个结点相连,构成链表。

链表结构的一大优势就是动态分配存储,不会像数组一样必须在定义时确定大小,造成不必要的浪费。用malloc和free函数即可实现开辟和释放存储单元。其中,malloc的参数多用sizeof运算符计算得到。

链表的基本操作有:正、反向建立链表;输出链表;删除链表中结点;在链表中插入结点等等,都是要熟练掌握的,初学者通过画图的方式能比较形象地理解建立、插入等实现的过程。

typedef struct nodeEDA中国门户网站,o0ah2w8ml,I
{
!h/Igu+Wa%M0char data;EDA中国门户网站|+j4ZkT5k!Boc;^$m
struct node *next;
'kp [4@0SyeP:\0}NODE; /*结点*/
Y e-\ fX3O#\ I0
n1H R| J2D"lE z_5unp0正向建立链表:EDA中国门户网站sr!Opi(tS&E8i(J c
NODE *create()EDA中国门户网站{ ^1s:LA\\
{
v5s#ZxW[ TJ0char ch='a';EDA中国门户网站$~2M2]8zM
NODE *p,*h=NULL,*q=NULL;
RAx/? L~b S0while(ch<'z')EDA中国门户网站:LL s%^'l/K
   {EDA中国门户网站l8z @v2d/y
   p=(NODE *)malloc(sizeof(NODE)); /*强制类型转换为指针*/EDA中国门户网站 I$shfubW
   p->data=ch;
P CV2L3y2A0   if(h==NULL) h=p;
]C:ivTP4w/o!zB0      else q->next=p;EDA中国门户网站;l} q2tfWQx
   ch++;EDA中国门户网站%? Ju8l*{
   q=p;
Irj)w_0   }
.q6G*Nr4aqqz0q->next=NULL; /*链表结束*/
b] fT9t'a0return h;EDA中国门户网站seN| Fn }
}

 

逆向建立:

NODE *create()EDA中国门户网站k o\ ~Zg%A^
{
$J$e)] BD_&I{!lj J0char ch='a';
l z&u!MJ\&say0NODE *p,*h=NULL;EDA中国门户网站:?1k3j$E|_X
while(ch<='z')
3q6z;j3} x4W G W@*WE0   {
`!y Sv%yF0   p=(NODE *)malloc(sizeof(NODE));EDA中国门户网站t6Y$Q1Z d
   p->data=ch;
4`5`:s4F8]0   p->next=h; /*不断地把head往前挪*/
MkvU\N)I"JYpU0   h=p;
^CE,Wc-i'K&L,n6p0   ch++;EDA中国门户网站{qQL.^C/h
   }
;?}o;N&^*T0return h;
5|F W$l$]aYw1O0}

 

用递归实现链表逆序输出:

void output(NODE *h)
R[5]&s,[0{
-F?7Ef:t0if(h!=NULL)
&PP J4\ jV`0   {
2Jv EMP5hw"f5z0   output(h->next);EDA中国门户网站M5hDIdNPX JIf
   printf("%c",h->data);EDA中国门户网站6hG8U;B#tS}&D'w
   }
E5u9?@m0N K+B9n0}

插入结点(已有升序的链表):

NODE *insert(NODE *h,int x)
7lh"@*n}H;v0{EDA中国门户网站| @V@E)g B-_*N
NODE *new,*front,*current=h;EDA中国门户网站9ej/\nFi8IU
while(current!=NULL&&(current->data<x)) /*查找插入的位置*/
j yT%WHJ0   {
-ps@%?[0D7c:d m0   front=current;
e+k4sY3L3\ A2{,]ox0   current=current->next;EDA中国门户网站rk(b\(`6T`D0}Gc
   }
wdGtJ.r&D2t V/d0new=(NODE *)malloc(sizeof(NODE));
!@"{vv%GeAb Flju0new->data=x;
{\(yubX_b[%p0new->next=current;EDA中国门户网站'oR*E'YT#t8YN
if(current==h) /*判断是否是要插在表头*/
"|8l/e \T0   h=new;
-kA SOd ~vN,N0else front->next=new;EDA中国门户网站}^ dHO\:]'w
return h;EDA中国门户网站'Qf l-]c]|$v
}

 

删除结点:

NODE *delete(NODE *h,int x)
\f7i"fsiS0{
L7?oKl4d9}\$J0NODE *q,*p=h;
;D/]8XL }Yj0while(p!=NULL&&(p->data!=x))
KU$odd%@tA+M0   {EDA中国门户网站` K|T!s fur ur
   q=p;
dAD3}-Fq0   p=p->next;
%wRV({}0Wc5t'j+h {0   }EDA中国门户网站;Mgv5^.^3g#tN
if(p->data==x) /*找到了要删的结点*/EDA中国门户网站sw{ j\#H4}m/~k
   {
!sfdS z*A0   if(p==h) /*判断是否要删表头*/EDA中国门户网站WV*aj9COvn
   h=h->next;EDA中国门户网站2q5vqJ$Ca R
      else q->next=p->next;EDA中国门户网站s\,t*{E
   free(p); /*释放掉已删掉的结点*/EDA中国门户网站o@%S'xL
   }EDA中国门户网站(fN]Ue},Se1b~H {G
return h;EDA中国门户网站5TP/n\xr/r:WD:Z V
}

 

经常有链表相关的程序填空题,做这样的题要注意看下面提到的变量是否定义了,用到的变量是否赋初值了,是否有给分配空间的没有分配空间,最后看看返回值是否正确。

笔者水平有限,难免有疏漏、错误的地方,浅显之处,还望指正见谅。上述内容仅是个提示作用,并不包括C语言的全部内容


FPGA/CPLD器件价格查询

TAG:

引用 删除 linch8247   /   2006-10-08 21:02:28
1
楼主有心了,你提醒了我,还是有必要复习一下C语言的,谢谢
 

评分:0

我来说两句

显示全部

:loveliness: :handshake :victory: :funk: :time: :kiss: :call: :hug: :lol :'( :Q :L ;P :$ :P :o :@ :D :( :)

日历

« 2008-11-10  
      1
2345678
9101112131415
16171819202122
23242526272829
30      

数据统计

  • 访问量: 14243
  • 日志数: 106
  • 图片数: 9
  • 文件数: 4
  • 建立时间: 2006-05-07
  • 更新时间: 2008-06-17

RSS订阅

Open Toolbar