本文对C语言的逻辑控制做一番较为深入的探讨,一般来说C语言的逻辑控制语句主要有如下的7种:
1、 goto 最强大,但一般只在特殊环境下使用。
2、 if else
3、 ?:
4、 switch case
5、 for
6、 while
7、 do while
自从dijkstra的论文Go To Statement Considered Harmful以后,C语言代码很少看到goto了。(一般用在多次资源分配的错误处理上)
但,从计算机的角度来说,缺少goto(jmp指令)还真没法干活。其实,goto是最符合我们的设计流程图的。
将一个流程图用goto来实现也最直观。goto能真正让我们做到心之所动,剑之所至。
接下来,分别考虑将2-7转化为对应的goto语言版本(等价于转化为对应汇编版本)。
2、if语句
1
2
3
4
|
if (条件1)
//代码块1
else
//代码块2
|
对应goto:
1
2
3
4
5
6
7
|
t = 条件1;
if (t为真) goto true ;
//代码块2
goto finish
true :
//代码块1
finish:
|
3、?: 和if else是等价的。
只不过if语句为代码块,?:为表达式。
1
|
变量= (条件1)? 表达式1:表达式2;
|
goto版本:
1
2
3
4
5
6
7
8
|
t = 条件1;
if (t为真)
goto true ;
//变量=表达式2
goto finish;
true :
//变量=表达式1
finish:
|
注意:?:里面的表达式尽量简单,过于复杂的话,改用if语句实现,这样方便调试。
4、switch-case
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
switch (条件取值变量)
case 元素1 :
//语句块1;
break ;
case 元素2:
//语句块2;
break ;
...
case 元素N
//语句块N;
break ;
default :
//默认处理。
break ;
}
|
goto版本:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
跳转表={标签1,标签2,...,标签N }
goto 跳转表[元素index]
//标签1:
//语句块1
goto Finish
//标签2:
//语句块2
goto Finish
...
//标签N:
//语句块N
goto Finish
default :
//默认处理。
Finish:
|
5、for语句
1
2
|
for (初始化语句;判断语句;迭代语句)
//循环语句块
|
goto版本:
1
2
3
4
5
6
7
8
9
|
//初始化语句;
if (判断语句为否)
goto Finish;
loop:
//循环语句块
//迭代语句
if (判断语句为真)
goto loop;
Finish:
|
6、while语句
1
2
3
|
while (条件为真)
//代码块
//迭代块
|
对应goto版本:
1
2
3
4
5
6
|
loop:
t = 条件
if (不为真) goto Finish;
//代码块
//迭代块
Finish:
|
7、do-while语句
1
2
3
4
|
do {
//语句块1
//迭代块1
} while (条件为真)
|
goto版本:
1
2
3
4
|
loop:
//语句块1
//迭代块1
if (条件为真) goto loop;
|
C语言的goto同汇编语句jmp系列指令逻辑上是完全一致的。
注意:
1、 关于条件,有个经典的逻辑代数公式:
摩根公式:
!(A && B) = (!A) || (!B)
建议,对复杂逻辑,手动画代数运算表
A B 结果
0 0 ?
0 1 ?
1 0 ?
1 1 ?
并保证单体测试的全覆盖。
2、 逻辑操作和位操作是2组,需要区分开来。
与 或 反
逻辑: && || !
位: & | ~ ^ (异或)
3 强烈推荐在写代码前,在纸上画完整的流程图,梳理自己的设计思路。