|
学习C语言的方法
/ g! N' _. N1 ]/ q+ s5 h) k( {: m- ^4 X0 }' R3 V" p" c* _
在初学C语言时,可能会遇到有些问题理解不透,或者表达方式与以往数学学习中不同(如运算符等),这就要求不气馁,不明白的地方多问多想,鼓足勇气进行学习,待学完后面的章节知识,前面的问题也就迎刃而解了,这一方面我感觉是我们同学最欠缺,大多学不好的就是因为一开始遇到困难就放弃,曾经和好多同学谈他的问题,回答是听不懂、不想听、放弃这样三个过程,我反问,这节课你听过课吗?回答又是没有,根本就没听过课,怎么说自己听不懂呢?相应的根本就没学习,又谈何学的好?
1 I8 B- k. \, r/ x. R7 W6 {' D, p& r
学习C语言始终要记住“曙光在前头”和“千金难买回头看”,“千金难买回头看”是学习知识的重要方法,就是说,学习后面的知识,不要忘了回头弄清遗留下的问题和加深理解前面的知识,这是我们学生最不易做到的,然而却又是最重要的。学习C语言就是要经过几个反复,才能前后贯穿,积累应该掌握的C知识。
+ p0 R4 S; f1 j2 \
' j( ]& s. }" N" }* n7 w 那么,我们如何学好《C程序设计》呢?
7 R. W) W, N) ?, J/ Q, ]
- L# i. ?5 p4 b% e: ~) W 一.学好C语言的运算符和运算顺序
8 b; P$ @3 ?: B: F0 c0 a. {. G# f* Z, c( Z% M) d( N
这是学好《C程序设计》的基础,C语言的运算非常灵活,功能十分丰富,运算种类远多于其它程序设计语言。在表达式方面较其它程序语言更为简洁,如自加、自减、逗号运算和三目运算使表达式更为简单,但初学者往往会觉的这种表达式难读,关键原因就是对运算符和运算顺序理解不透不全。当多种不同运算组成一个运算表达式,即一个运算式中出现多种运算符时,运算的优先顺序和结合规则显得十分重要。在学习中,只要我们对此合理进行分类,找出它们与我们在数学中所学到运算之间的不同点之后,记住这些运算也就不困难了,有些运算符在理解后更会牢记心中,将来用起来得心应手,而有些可暂时放弃不记,等用到时再记不迟。 ( {, ?) e X9 F$ ]3 ^& F$ |
: N+ K, S8 p8 i 先要明确运算符按优先级不同分类,《C程序设计》运算符可分为15种优先级,从高到低,优先级为1 ~ 15,除第2、13级和第14级为从右至左结合外,其它都是从左至右结合,它决定同级运算符的运算顺序. - W$ A% ?$ i E$ k7 n6 k" Z/ v
* {' Q2 b" k/ t
二.学好C语言的四种程序结构 / B0 Q' ?7 P) {( k& m$ Q
w: ]. l8 t- }3 Z
(1)顺序结构
/ U3 d5 A$ L( B: V: N
* X' B. D. b8 U' G; G 顺序结构的程序设计是最简单的,只要按照解决问题的顺序写出相应的语句就行,它的执行顺序是自上而下,依次执行。 b3 X4 g4 ?- H' I0 a' r: z
' T" l6 D8 Y0 N3 _9 K2 k, N( k5 J
例如;a = 3,b = 5,现交换a,b的值,这个问题就好像交换两个杯子水,这当然要用到第三个杯子,假如第三个杯子是c,那么正确的程序为: c = a; a = b; b = c; 执行结果是a = 5,b = c = 3如果改变其顺序,写成:a = b; c = a; b = c; 则执行结果就变成a = b = c = 5,不能达到预期的目的,初学者最容易犯这种错误。 顺序结构可以独立使用构成一个简单的完整程序,常见的输入、计算,输出三步曲的程序就是顺序结构,例如计算圆的面积,其程序的语句顺序就是输入圆的半径r,计算s = 3.14159*r*r,输出圆的面积s。不过大多数情况下顺序结构都是作为程序的一部分,与其它结构一起构成一个复杂的程序,例如分支结构中的复合语句、循环结构中的循环体等。 * q8 i+ ^ O X) M" H! }
+ W; ]9 _; L2 b8 ~' m
(2) 分支结构 / k4 a4 u* J3 b- \6 S
, a3 P$ Z" `# ?& n9 E" K2 a4 r
顺序结构的程序虽然能解决计算、输出等问题,但不能做判断再选择。对于要先做判断再选择的问题就要使用分支结构。分支结构的执行是依据一定的条件选择执行路径,而不是严格按照语句出现的物理顺序。分支结构的程序设计方法的关键在于构造合适的分支条件和分析程序流程,根据不同的程序流程选择适当的分支语句。分支结构适合于带有逻辑或关系比较等条件判断的计算,设计这类程序时往往都要先绘制其程序流程图,然后根据程序流程写出源程序,这样做把程序设计分析与语言分开,使得问题简单化,易于理解。程序流程图是根据解题分析所绘制的程序执行流程图。 . n/ T Z4 x/ K; u6 N0 _& F
' z& b" K& u; e/ U3 E' e# @7 P/ e& W& I 学习分支结构不要被分支嵌套所迷惑,只要正确绘制出流程图,弄清各分支所要执行的功能,嵌套结构也就不难了。嵌套只不过是分支中又包括分支语句而已,不是新知识,只要对双分支的理解清楚,分支嵌套是不难的。下面我介绍几种基本的分支结构。
; l2 e3 F6 Q3 a: k: {& A/ U \1 ]& A& R$ w: p& V
①if(条件) 5 W' r" p# Y# F$ p+ G/ ]
2 z& U5 @' ~8 E; x b
{分支体} 6 U1 I; x3 k, x$ a4 y! C: i! S- g! j
5 N3 E4 x6 \8 T+ d* e: S
这种分支结构中的分支体可以是一条语句,此时“{ }”可以省略,也可以是多条语句即复合语句。它有两条分支路径可选,一是当条件为真,执行分支体,否则跳过分支体,这时分支体就不会执行。如:要计算x的绝对值,根据绝对值定义,我们知道,当x>=0时,其绝对值不变,而x<0时其绝对值是为x的反号,因此程序段为:if(x<0) x=-x; - `( n- G- L7 I) e( ?5 k( s
8 O: a! r, `% @' x3 H) w4 q ②if(条件)
: N# P; v. ]$ I; c8 B6 n2 `3 G, T( e4 S7 t
{分支1} ' D- ?! r l* s+ F' I; c. I9 p0 ^
$ I b4 b/ L2 q9 J
else
+ p! w" U3 f) Q+ T
! T; o \; ?; B. w+ o; m {分支2}
; m- V. G0 B6 F f7 S# E
- H) _1 G$ V' [4 O7 B 这是典型的分支结构,如果条件成立,执行分支1,否则执行分支2,分支1和分支2都可以是1条或若干条语句构成。如:求ax^2+bx+c=0的根 X% U) s6 l5 R' k
4 c% P) q6 P" v( Q, Q
分析:因为当b^2-4ac>=0时,方程有两个实根,否则(b^2-4ac<0)有两个共轭复根。其程序段如下: - U) l4 v6 C$ N E6 B- u, ~9 V
" {+ V8 {# O( k d=b*b-4*a*c;
$ O6 }/ P, p8 l% e" X% b/ x; g! B2 u6 _: }
if(d>=0)
/ l$ j# D$ {- S- ~' y! c
, v- X5 M, D! b) c9 y+ N2 P {x1=(-b+sqrt(d))/2a; ! B8 z5 V) u0 H
$ \9 L+ q- ~1 V; p5 D L3 h
x2=(-b-sqrt(d))/2a; ! J6 m! U! P4 C S/ W/ V: Q+ h- r) u
! S8 {/ h* x2 O+ X* l7 {6 j printf(“x1=%8.4f,x2=%8.4f\n”,x1,x2); % _ I, M8 H& i
/ N5 z+ l" {" f* t+ H
} ! O5 ?1 j# Y( I7 Z$ _' o! C! I& h
1 f* N. m# e# {" c0 _. l$ Q' H8 D1 [ else
0 T# i0 ?5 ~* Q* A
4 n/ s. c& k! n" v+ u {r=-b/(2*a);
; m8 p+ ]/ r+ V2 m* x- q, m8 C* E) m) O7 L
i =sqrt(-d)/(2*a); 4 z) G: n' O& D9 ~- A4 K
) O9 y$ w6 |2 \' z, p2 a printf(“x1=%8.4f+%8.4fi\n”r, i); ' Y6 Z3 V% V- T# d8 Z2 [
7 W# r, ~+ K/ ]! }/ |2 k printf(“x2=%8.4f-%8.4fi\n”r,i) $ I4 V3 B6 D$ }! u k
8 {. ~$ i3 G+ ?8 _0 W. j/ Y# t4 }* V }
! N0 m+ w4 K6 i" U' q1 r/ Z; ^: g: ~7 n' j
③嵌套分支语句:其语句格式为:
6 s2 l$ A$ a' o( h% ~1 c7 B' h t( B3 M2 [3 `# I
if(条件1) {分支1}; ( a6 R' F9 z* Q# G2 Y2 R$ {0 X
0 o$ \' M _6 G" i0 W8 }9 j else if(条件2) {分支2}
7 n! D; V3 \( x0 r6 S# Y# W" ^* P, N8 ^3 `9 f9 e" e
else if(条件3) {分支3}
; x# x3 b1 J5 I3 x8 u* t& n' F/ G6 ` D& g/ C p
……
" S/ y1 a: _8 S4 h6 P+ }& g' N2 n; r- P/ ?
else if(条件n) {分支n}
0 m% }/ J, a: e! L8 s5 |1 w# y6 n5 @: D
else {分支n+1} * Z0 d. J! Z4 \! _0 _# G
9 A/ ~& n% E" L; R+ `5 u' h
嵌套分支语句虽可解决多个入口和出口的问题,但超过3重嵌套后,语句结构变得非常复杂,对于程序的阅读和理解都极为不便,建议嵌套在3重以内,超过3重可以用下面的语句。
4 b( \/ m, l+ z3 P8 `0 p1 J! ~, A4 S+ y q, r% }6 d
④switch开关语句:该语句也是多分支选择语句,到底执行哪一块,取决于开关设置,也就是表达式的值与常量表达式相匹配的那一路,它不同if…else 语句,它的所有分支都是并列的,程序执行时,由第一分支开始查找,如果相匹配,执行其后的块,接着执行第2分支,第3分支……的块,直到遇到break语句;如果不匹配,查找下一个分支是否匹配。这个语句在应用时要特别注意开关条件的合理设置以及break语句的合理应用。 ' t, H4 s' A; U }: O
6 w; G: w; A: Q& e3 G, D! z( U
(3)循环结构: " E t$ {, E$ |
! d: `# A- c; ^( D% B% w' N+ J- ? 循环结构可以减少源程序重复书写的工作量,用来描述重复执行某段算法的问题,这是程序设计中最能发挥计算机特长的程序结构,C语言中提供四种循环,即goto循环、while循环、do ?Cwhile循环和for循环。四种循环可以用来处理同一问题,一般情况下它们可以互相代替换,但一般不提倡用goto循环,因为强制改变程序的顺序经常会给程序的运行带来不可预料的错误,在学习中我们主要学习while、do…while、for三种循环。常用的三种循环结构学习的重点在于弄清它们相同与不同之处,以便在不同场合下使用,这就要清楚三种循环的格式和执行顺序,将每种循环的流程图理解透彻后就会明白如何替换使用,如把while循环的例题,用for语句重新编写一个程序,这样能更好地理解它们的作用。特别要注意在循环体内应包含趋于结束的语句(即循环变量值的改变),否则就可能成了一个死循环,这是初学者的一个常见错误。
; I4 G4 y* W1 S0 g" r$ s" \8 ?! ^/ m& M
在学完这三个循环后,应明确它们的异同点:用while和do…while循环时,循环变量的初始化的操作应在循环体之前,而for循环一般在语句1中进行的;while 循环和for循环都是先判断表达式,后执行循环体,而do…while循环是先执行循环体后判断表达式,也就是说do…while的循环体最少被执行一次,而while 循环和for就可能一次都不执行。另外还要注意的是这三种循环都可以用break语句跳出循环,用continue语句结束本次循环,而goto语句与if构成的循环,是不能用break和 continue语句进行控制的。
$ i) m& m" A( d2 U$ r: `" B2 C" |! A4 x" [/ g) k" } x
顺序结构、分支结构和循环结构并不彼此孤立的,在循环中可以有分支、顺序结构,分支中也可以有循环、顺序结构,其实不管哪种结构,我们均可广义的把它们看成一个语句。在实际编程过程中常将这三种结构相互结合以实现各种算法,设计出相应程序,但是要编程的问题较大,编写出的程序就往往很长、结构重复多,造成可读性差,难以理解,解决这个问题的方法是将C程序设计成模块化结构。 " Q6 F$ I8 B) G P
+ m7 G0 r8 |7 E
(4)模块化程序结构
7 t9 _0 ?1 u3 }+ a2 c8 r! F! v. a6 x, I _7 o
C语言的模块化程序结构用函数来实现,即将复杂的C程序分为若干模块,每个模块都编写成一个C函数,然后通过主函数调用函数及函数调用函数来实现一大型问题的C程序编写,因此常说:C程序=主函数+子函数。 因此,对函数的定义、调用、值的返回等中要尤其注重理解和应用,并通过上机调试加以巩固。
) Y( i6 {* p. d7 j c r
K6 Y5 W% `2 r1 U7 K2 \ 三.掌握一些简单的算法 / e, W/ C% E; \* O2 {4 D" Q4 {2 h
* D" \- \% Q5 ^3 o' p: j) ?
编程其实一大部分工作就是分析问题,找到解决问题的方法,再以相应的编程语言写出代码。这就要求掌握算法,根据我们的《C程序设计》教学大纲中,只要求我们掌握一些简单的算法,在掌握这些基本算法后,要完成对问题的分析就容易了。如两个数的交换、三个数的比较、选择法排序和冒泡法排序,这就要求我们要清楚这些算法的内在含义
0 ` k j; ?+ |$ @ e% U! V' s6 l% V' J- E6 D0 R
结语:当我们把握好上述几方面后,只要同学们能克服畏难、厌学、上课能专心听讲,做好练习与上机调试,其实C语言并不难学 9 \7 r/ ] W. j1 z0 o1 C) e
c; j% o9 h! W p+ G
C源程序的关键字---------------------------------------------------------------------------------------
8 S5 T) B4 e; `5 h1 ~' R( _4 v' [- y; v. n" Q
所谓关键字就是已被C语言本身使用, 不能作其它用途使用的字。例如关键字不能用作变量名、函数名等
% Y" P, X m9 y, l/ O2 D# w0 D4 Q; T* B% U2 j b4 x
由ANSI标准定义的C语言关键字共32个 :
" @; G. d/ j8 {* G: T- w/ z9 Z) j; O' l: a5 S* p0 I
auto double int struct break else long switch 8 ~4 |& d" Y$ {$ m) `
( V' x/ h% l$ j! E case enum register typedef char extern return union 2 T. m& A$ t/ D' b$ ?( _
0 g0 S' {( `9 e: k v( E! V8 x4 t const float short unsigned continue for signed void 4 s& P ~$ Z3 b
/ Q0 j# r( {. B
default goto sizeof volatile do if while static # f! m8 X! l* |4 j
2 [$ A# l+ |. E F2 \9 \: e* j 根据关键字的作用,可以将关键字分为数据类型关键字和流程控制关键字两大类。 ( v0 s; r- G3 p; ^
& u+ t1 B+ O& r B 1 数据类型关键字
% ?9 B9 c- Z+ s2 V( D. l
: h1 [6 Y1 s: z3 y) }7 P* Q A基本数据类型(5个) + S. I+ @& C- H; A
1 Q+ Z* F6 H6 ^! v/ I. G6 {* \
void :声明函数无返回值或无参数,声明无类型指针,显式丢弃运算结果 , N% n3 b8 \+ A- z5 C% s
! Y) q: \! z2 s2 I char :字符型类型数据,属于整型数据的一种
+ i+ [4 A$ l0 R& l% m2 `5 M; M
) r+ S5 a& Y+ _0 q8 M int :整型数据,通常为编译器指定的机器字长
0 f. j7 F/ d$ ~. n
0 d1 T j. X5 B; j" | float :单精度浮点型数据,属于浮点数据的一种 . T* k3 S0 q5 P7 q; K. u" T
1 X2 B; D) y9 [/ F
double :双精度浮点型数据,属于浮点数据的一种
7 \ _) P/ R* A$ E9 S0 R
! R) `6 z% y- J' {4 } B 类型修饰关键字(4个) / z7 k6 e) y5 h+ S, Q: S" y
: @. j' n" j' p. B short :修饰int,短整型数据,可省略被修饰的int。 8 r5 m' y% ~! U. }
8 |- D( x5 u: N- q8 } long :修饰int,长整形数据,可省略被修饰的int。 + C1 V7 y) h" V" y6 ?$ ~
5 {* I" o6 r6 J
signed :修饰整型数据,有符号数据类型 9 P% M( s2 [% i J" E$ b* @% Q
& O, h X( u$ u! n) D' w+ a2 x unsigned :修饰整型数据,无符号数据类型 ' m+ K" n. I0 x7 [2 [2 s ^
7 c3 G: [" A3 D! k7 O% p7 B
C 复杂类型关键字(5个) 9 ?, {) z, r, o" Y; V. y; f: E b0 ?
' U) f! n v- s
struct :结构体声明 8 q a- q! K: t9 W+ @
1 `$ g6 _# P: F, r8 V$ W union :共用体声明
8 n, R/ F% y g7 y0 {
: n4 p* @% |) Q9 ]" Q enum :枚举声明 8 |7 ~1 }. c- K/ U1 h) z
! R! }# F w. o typedef :声明类型别名
6 O+ y6 F: O/ g4 a0 }
0 H! Z5 C: m7 z sizeof :得到特定类型或特定类型变量的大小 7 N" ?$ L2 ?) v: q+ }
2 `+ A) e! u3 j! h9 a
D 存储级别关键字(6个) + [/ v$ t+ C9 _. ^8 K: s% w
3 X8 P! }; c; x" f: x; T
auto :指定为自动变量,由编译器自动分配及释放。通常在栈上分配 5 z) f% s8 `+ Z. ]' ^+ S+ }
. n! m" r3 \5 z, Z9 G! B* _ static :指定为静态变量,分配在静态变量区,修饰函数时,指定函数作用域为文件内部
) S/ A3 L# G3 [$ [3 f2 y1 r* p- L
register :指定为寄存器变量,建议编译器将变量存储到寄存器中使用,也可以修饰函数形参,建议编译器通过寄存器而不是堆栈传递参数 8 q% b8 c+ j2 j4 J
4 E- F( k9 b1 R5 T+ v extern :指定对应变量为外部变量,即标示变量或者函数的定义在别的文件中,提示编译器遇到此变量和函数时在其他模块中寻找其定义。 8 Z3 z6 B( k- l
% g$ K4 q& T5 E* | const :与volatile合称“cv特性”,指定变量不可被当前线程/进程改变(但有可能被系统或其他线程/进程改变)
+ K0 Y) o7 v* p6 B0 q$ n
7 g; G3 i8 J3 H, L volatile :与const合称“cv特性”,指定变量的值有可能会被系统或其他进程/线程改变,强制编译器每次从内存中取得该变量的值
# a4 R, b" U+ W9 ?) b6 J0 s, L$ u: F* [
2 流程控制关键字 * o! J8 u, G0 Z/ r
6 D: H. f. F4 @% o) r( [# _
A 跳转结构(4个)
' ~ Y2 x; ]& P: |- L
8 f) G! B7 g ` return :用在函数体中,返回特定值(或者是void值,即不返回值) , t J) g1 X6 \0 `/ c j2 U
# k+ y* s$ c: M
continue :结束当前循环,开始下一轮循环 ; T z e. A ]9 s& m2 f
' w0 n- n9 I- }+ c1 Q! Z) e break :跳出当前循环或switch结构
# n5 V. U$ b5 U% w- Z: z/ ^ b; u: F2 `: W
goto :无条件跳转语句 ( t/ I& E! }0 z, u4 L4 v$ D
* U, K4 _( l$ ]" `
B 分支结构(5个)
* U1 x1 e& O/ h8 [/ F" D$ O! _8 E4 V
if :条件语句 4 W' G. u7 L! G# h4 c" k
8 b+ [# v- P" I else :条件语句否定分支(与if连用)
7 E7 w; y- t; {1 s" H- J8 W, b) P, F
switch :开关语句(多重分支语句) 0 t+ m0 H( F6 \% r+ I8 }
1 v1 n& F5 x. a$ x- F7 {+ `6 e' J+ u
case :开关语句中的分支标记 1 X% r5 \( \7 Z' g
% N! \) p- y7 Y0 n" e default :开关语句中的“其他”分治,可选。
: J" I- F2 L6 z3 u" ~
1 _% t9 w$ e, { C 循环结构(3个) 6 _& g. a9 Y+ u" y. j! x
" o! X& u+ ?! j8 f1 d% v+ a$ ? for :for循环结构,for(1;2;3)4;的执行顺序为1->2->4->3->2...循环,其中2为循环条件 . D5 h8 I* D1 O s$ e' s" Z& Q
2 |: {, }; K0 o5 B% n5 N
do :do循环结构,do 1 while(2); 的执行顺序是 1->2->1...循环,2为循环条件
- [ ~/ i* @& E2 _2 y# w& z) F P+ O, M7 Z& ]
while :while循环结构,while(1) 2; 的执行顺序是1->2->1...循环,1为循环条件
1 K& H. C# q! j3 v3 ]2 t8 }/ ]
1 W9 W8 Z0 O7 p, Z' i% { [ 以上循环语句,当循环条件表达式为真则继续循环,为假则跳出循环。 |
评分
-
查看全部评分
|