新书推荐:6.2 else if语句

本节必须掌握的知识点:

  示例代码二十

   代码分析

    汇编解析

if语句表达形式3

if(表达式1)

      statement1

else if(表达式2)

statement2

else if(表达式3)

statement3

……

else

  statementN

解析:

如果表达式1非0,则执行statement1,执行完退出语句;

如果表达式2非0,则执行statement2,执行完退出语句;

如果表达式3非0,则执行statement3,执行完退出语句;

如果表达式4非0,则执行statement4,执行完退出语句;

……

如果所有的表达式都不满足,则执行else对应的statement语句。

示例代码二十

6.2.1 示例二十

●第一步:分析需求,设计程序结构框架。

分析需求:某中学考试成绩分为ABCDE 5个等级,在[90,100]区间内的成绩为A,在[80,90)区间内的成绩为B,在[70,80)区间内的成绩为C,在[60,70)区间内的成绩为D,在[0,60)区间内的成绩为E。

设计程序结构框架:分支结构(if语句形式3)if/else if语句。

       ●第二步:数据定义,定义恰当的数据结构;

       int score;//定义一个int类型的整型局部变量。

       ●第三步:分析算法。

       根据条件表达式判断考试分数属于哪个等级。

       ●第四步:编写伪代码,即用我们自己的语言来编写程序。

       int main(void) {

    定义一个int类型整型变量num;

    调用printf函数打印一个提示信息"请输入一个整数:";

    调用scanf_s函数接收键盘输入一个整数,并存入变量num;

    if (score >= 90 && score <= 100)如果条件为真

        调用printf函数输出"您在A等级\n";

else if (score >= 80 && score < 90) 如果条件为真

        调用printf函数输出"您在B等级\n";

else if (score >= 70 && score < 80) 如果条件为真

        调用printf函数输出"您在C等级\n";

else if (score >= 60 && score < 70) 如果条件为真

        调用printf函数输出"您在D等级\n";

else 其他情形

        调用printf函数输出"您在E等级\n";

    system("pause");

    return 0;                                            

}

●第五步:画流程图,使用Visio、Excel或者其他绘图工具绘制算法流程和逻辑关系图;    

             

图6-3 示例二十流程图

●第六步:编写源程序,其实就是将我们的伪代码翻译成计算机语言;

/*

   根据学生考试分数,输出ABCDE 5个等级

*/

#include <stdio.h>

#include <stdlib.h>

int main(void) {

    int score = 0;

    printf("请输入您的成绩:");

    scanf_s("%d", &score);

    if (score >= 90 && score <= 100)

        printf("您在A等级\n");

    else if (score >= 80 && score < 90)

        printf("您在B等级\n");

    else if (score >= 70 && score < 80)                            

        printf("您在C等级\n");

    else if (score >= 60 && score < 70)                   

        printf("您在D等级\n");

    else

        printf("您在E等级\n");

    system("pause");

    return 0;

}

输出结果:

请输入您的成绩:91

您在A等级

       请输入您的成绩:81

您在B等级

请输入您的成绩:71

您在C等级

请输入您的成绩:61

您在D等级

请输入您的成绩:0

您在E等级

       ●第七步:调试程序,修复程序中可能出现的BUG;

参见反汇编代码。

●第八步:优化代码,尝试更好的设计方案,效率更高的算法,逻辑更为清晰简洁明了。

示例二十一中我们将改用if/else结构的简化形式switch结构。

6.2.2 代码分析                     

解析执行过程:

  1. 输入成绩,score = 71;虽然成绩通常不会是一个负整数,应该定义一个usigned int类型变量,这是对的。但是int类型可以表示的数据范围足够使用,所以这里并没有定义为usigned int类型,而是int类型也没什么问题。
  2. 执行表达式(score>=90 && score<=100),值为0,继续判断其他条件;
  3. 执行表达式(score>=80 && score<90),值为0,继续判断其他条件;

4、执行表达式(score>=70 && score<80),值为0,满足条件,执行对应的语句: printf("您在C等级\n"); 跳出if语句,不再判断其它if语句。

5、结束。

   【注意】为了增强程序的可读性强,我们通常会保留大括号,如下所示:

#include <stdio.h>

#include <windows.h>

int main(void)

{

       int score = 0;

       printf("请输入您的成绩:");

加上了程序界定符“{” “}” ,这样增强代码的可读性

       scanf("%d",&score);

       if(score>=90 && score<=100)

       {

         printf("您在A等级\n");

       }

       else if(score>=80 && score<90)

       {

         printf("您在B等级\n");

       }

       else if(score>=70 && score<80)

       {

         printf("您在C等级\n");

       }

       else if(score>=60 && score<70)

       {

         printf("您在D等级\n");

       }

       else

       {

         printf("您在E等级\n");

       }

    system("pause");

    return 0;

}

6.2.3 汇编解析

汇编代码

;C标准库头文件和导入库

include vcIO.inc

.data      

score sdword  ?

.const    

szMsg1 db "请输入您的成绩:",0

szMsg2 db "%d",0

szMsg3 db "您在A等级",0dh,0ah,0

szMsg4 db "您在B等级",0dh,0ah,0

szMsg5 db "您在C等级",0dh,0ah,0

szMsg6 db "您在D等级",0dh,0ah,0

szMsg7 db "您在E等级",0dh,0ah,0

.code     

start:

       ;输入整数num

       invoke printf,offset szMsg1

       invoke scanf,offset szMsg2,ADDR score

       ;

       .if (score >= 90 && score <= 100)

              invoke printf,offset szMsg3

       .elseif (score >= 80 && score < 90)

              invoke printf,offset szMsg4

       .elseif (score >= 70 && score < 80)

              invoke printf,offset szMsg5        

       .elseif (score >= 60 && score < 70)

              invoke printf,offset szMsg6        

       .else

              invoke printf,offset szMsg7 

       .endif

       ;     

       invoke _getch

       ret                       

end start

       上述汇编代码使用了高级汇编伪指令.if/.elseif/.else,竟然与C语言的语法如此的相似,很难不让人怀疑,C语言的发明人是否借鉴了高级汇编的语法。当然我们也可以理解为什么有人会将C语言称为高级汇编语言了。

       为了彻底搞清楚计算机是如何实现这些分支结构的条件判断,我们还是看一看反汇编代码,了解计算机的具体实现过程。

反汇编代码

           int score = 0;

00B51952  mov         dword ptr [score],0 

    printf("请输入您的成绩:");

00B51959  push        offset string "\xc7\xeb\xca\xe4\xc8\xeb\xc4\xfa\xb5\xc4\xb3\xc9\xbc\xa8\xa3\xba" (0B57B30h) 

00B5195E  call        _printf (0B5104Bh) 

00B51963  add         esp,4 

    scanf_s("%d", &score);

00B51966  lea         eax,[score] 

00B51969  push        eax 

00B5196A  push        offset string "%d" (0B57B44h) 

00B5196F  call        _scanf_s (0B51154h) 

00B51974  add         esp,8 

    if (score >= 90 && score <= 100)

00B51977  cmp         dword ptr [score],5Ah ;比较score和90的大小

00B5197B  jl          main+72h (0B51992h)  ;如果score<90则跳转到下一个else if语句

00B5197D  cmp         dword ptr [score],64h ;比较score和100的大小

00B51981  jg          main+72h (0B51992h)   ;如果score>100则跳转到下一个else if语句

        printf("您在A等级\n") ;90<=score<=100则执行下面的printf语句

00B51983  push        offset string "\xc4\xfa\xd4\xdaA\xb5\xc8\xbc\xb6\n" (0B57B48h) 

00B51988  call        _printf (0B5104Bh) 

00B5198D  add         esp,4 

00B51990  jmp         main+0D0h (0B519F0h) 

    else if (score >= 80 && score < 90)

00B51992  cmp         dword ptr [score],50h 

00B51996  jl          main+8Dh (0B519ADh)  ;jl小于

00B51998  cmp         dword ptr [score],5Ah 

00B5199C  jge         main+8Dh (0B519ADh)  ;jg大于等于

        printf("您在B等级\n");

00B5199E  push        offset string "\xc4\xfa\xd4\xdaB\xb5\xc8\xbc\xb6\n" (0B57B58h) 

00B519A3  call        _printf (0B5104Bh) 

00B519A8  add         esp,4 

00B519AB  jmp         main+0D0h (0B519F0h) 

    else if (score >= 70 && score < 80)

00B519AD  cmp         dword ptr [score],46h 

00B519B1  jl          main+0A8h (0B519C8h) 

00B519B3  cmp         dword ptr [score],50h 

00B519B7  jge         main+0A8h (0B519C8h) 

        printf("您在C等级\n");

00B519B9  push        offset string "\xc4\xfa\xd4\xdaC\xb5\xc8\xbc\xb6\n" (0B57B68h) 

        printf("您在C等级\n");

00B519BE  call        _printf (0B5104Bh) 

00B519C3  add         esp,4 

00B519C6  jmp         main+0D0h (0B519F0h) 

    else if (score >= 60 && score < 70)

00B519C8  cmp         dword ptr [score],3Ch 

00B519CC  jl          main+0C3h (0B519E3h) 

00B519CE  cmp         dword ptr [score],46h 

00B519D2  jge         main+0C3h (0B519E3h) 

        printf("您在D等级\n");

00B519D4  push        offset string "\xc4\xfa\xd4\xdaD\xb5\xc8\xbc\xb6\n" (0B57B78h) 

00B519D9  call        _printf (0B5104Bh) 

00B519DE  add         esp,4 

00B519E1  jmp         main+0D0h (0B519F0h) 

    else

        printf("您在E等级\n");

00B519E3  push        offset string "\xc4\xfa\xd4\xdaE\xb5\xc8\xbc\xb6\n" (0B57B88h) 

00B519E8  call        _printf (0B5104Bh) 

00B519ED  add         esp,4 

       由上述反汇编代码可知,C语言的if/else if分支结构由一系列的cmp/jcc条件判断指令实现的。例如:

           if (score >= 90 && score <= 100)

00B51977  cmp         dword ptr [score],5Ah ;比较score和90的大小

00B5197B  jl          main+72h (0B51992h)  ;如果score<90则跳转到下一个else if语句

00B5197D  cmp         dword ptr [score],64h ;比较score和100的大小

00B51981  jg          main+72h (0B51992h)   ;如果score>100则跳转到下一个else if语句

        printf("您在A等级\n") ;90<=score<=100则执行下面的printf语句

00B51983  push        offset string "\xc4\xfa\xd4\xdaA\xb5\xc8\xbc\xb6\n" (0B57B48h) 

00B51988  call        _printf (0B5104Bh) 

00B5198D  add         esp,4 

00B51990  jmp         main+0D0h (0B519F0h) 

    else if (score >= 80 && score < 90)

00B51992  cmp         dword ptr [score],50h 

 

结论

C语言中的if/else if语句就是一系列CMP/JCC指令的连续判断,大大简化了代码。反汇编代码就是编译器将C语言翻译后的汇编代码。

实验三十六:跟踪调试示例程序

第一步:打开DtDebug调试器,将示例二十生成的MyProjectOne.exe程序拖入调试器。

第二步:按Ctrl+F9,进入程序入口地址,如图6-4所示。

图6-4 进入程序入口地址

       第三步:按F8,跳转到0x00c92380地址处,发现只有一个call语句,如图6-5所示。连续按3次F7或者直接选中call语句,点击鼠标右键,选中fllow跟随,进入call函数内部。

图6-5 F7单步步入函数内

       第四步:进入call函数内部,如果6-6所示,此函数内有两个call语句。先按F8单步步过测试一下,第一个call没有反应,执行完第二个call语句时,控制台窗口显示提示信息:

“请输入您的成绩:”。

图6-6 F8 call测试

 第五步:点击左上角工具栏

图标,重新来过,重复前面的1,2,3步,第4步至第二个call语句时,改用F7单步步入,如图6-7所示。

图6-7 F7单步步入第二个call函数   

图6-8 锁定程序代码段的真实开始地址处

第六步:继续按F8测试,直至控制台窗口出现提示信息“请输入您的成绩:”。锁定地址0x00c92172地址处的call语句(每次调试加载程序的地址会有不同),鼠标选中后按F2,这该地址处下软件断点。如图6-8所示。

第七步:再次重新加载调试程序,连续点击两次左上角工具栏

图标执行程序,在0x00c92172地址处断下。然后按F7单步步入。

第八步:继续按F8测试,锁定最后一个call语句输出提示信息。然后在地址0x00c92309地址处按F2下断点。如图6-9所示。

图6-9 锁定最终的入口函数

第九步:重新来过,连续执行至0x00c92309地址处,按F7单步步入,然后再按一次F7跳转到main函数内,如下所示:

建立堆栈框架

00C91920    55              PUSH EBP

00C91921    8BEC            MOV EBP,ESP

00C91923    81EC D0000000   SUB ESP,0D0

保护寄存器

00C91929    53              PUSH EBX

00C9192A    56              PUSH ESI

00C9192B    57              PUSH EDI

00C9192C    8DBD 30FFFFFF   LEA EDI,DWORD PTR SS:[EBP-D0]

初始化堆栈

00C91932    B9 34000000     MOV ECX,34

00C91937    B8 CCCCCCCC     MOV EAX,CCCCCCCC

00C9193C    F3:AB           REP STOS DWORD PTR ES:[EDI]

00C9193E    A1 04A0C900     MOV EAX,DWORD PTR DS:[C9A004]

00C91943    33C5            XOR EAX,EBP

00C91945    8945 FC         MOV DWORD PTR SS:[EBP-4],EAX

00C91948    B9 03C0C900     MOV ECX,MyProjec.00C9C003

00C9194D    E8 CFF8FFFF     CALL MyProjec.00C91221

00C91952    C745 F4 0000000>MOV DWORD PTR SS:[EBP-C],0

以上00C9193E~00C91952地址处的代码为编译器添加的校验代码,此处不做介绍。

printf("请输入您的成绩:");

printf函数

00C91959    68 307BC900     PUSH MyProjec.00C97B30

00C9195E    E8 E8F6FFFF     CALL MyProjec.00C9104B

00C91963    83C4 04         ADD ESP,4

00C91966    8D45 F4         LEA EAX,DWORD PTR SS:[EBP-C]

scanf_s("%d", &score);

scanf函数

00C91969    50              PUSH EAX

00C9196A    68 447BC900     PUSH MyProjec.00C97B44 ; ASCII "%d"

00C9196F    E8 E0F7FFFF     CALL MyProjec.00C91154

00C91974    83C4 08         ADD ESP,8

if (score >= 90 && score <= 100)

00C91977    837D F4 5A      CMP DWORD PTR SS:[EBP-C],5A

If语句条件判断

00C9197B    7C 15           JL SHORT MyProjec.00C91992

00C9197D    837D F4 64      CMP DWORD PTR SS:[EBP-C],64

00C91981    7F 0F           JG SHORT MyProjec.00C91992

printf("您在A等级\n");

printf函数

00C91983    68 487BC900     PUSH MyProjec.00C97B48

00C91988    E8 BEF6FFFF     CALL MyProjec.00C9104B

00C9198D    83C4 04         ADD ESP,4

00C91990    EB 5E           JMP SHORT MyProjec.00C919F0

else if (score >= 80 && score < 90)

00C91992    837D F4 50      CMP DWORD PTR SS:[EBP-C],50

00C91996    7C 15           JL SHORT MyProjec.00C919AD

00C91998    837D F4 5A      CMP DWORD PTR SS:[EBP-C],5A

00C9199C    7D 0F           JGE SHORT MyProjec.00C919AD

printf("您在B等级\n");

00C9199E    68 587BC900     PUSH MyProjec.00C97B58

00C919A3    E8 A3F6FFFF     CALL MyProjec.00C9104B

00C919A8    83C4 04         ADD ESP,4

00C919AB    EB 43           JMP SHORT MyProjec.00C919F0

else if (score >= 70 && score < 80)

00C919AD    837D F4 46      CMP DWORD PTR SS:[EBP-C],46

00C919B1    7C 15           JL SHORT MyProjec.00C919C8

00C919B3    837D F4 50      CMP DWORD PTR SS:[EBP-C],50

00C919B7    7D 0F           JGE SHORT MyProjec.00C919C8

printf("您在C等级\n");

00C919B9    68 687BC900     PUSH MyProjec.00C97B68

00C919BE    E8 88F6FFFF     CALL MyProjec.00C9104B

00C919C3    83C4 04         ADD ESP,4

00C919C6    EB 28           JMP SHORT MyProjec.00C919F0

else if (score >= 60 && score < 70)

00C919C8    837D F4 3C      CMP DWORD PTR SS:[EBP-C],3C

00C919CC    7C 15           JL SHORT MyProjec.00C919E3

00C919CE    837D F4 46      CMP DWORD PTR SS:[EBP-C],46

00C919D2    7D 0F           JGE SHORT MyProjec.00C919E3

printf("您在D等级\n");

00C919D4    68 787BC900     PUSH MyProjec.00C97B78

00C919D9    E8 6DF6FFFF     CALL MyProjec.00C9104B

00C919DE    83C4 04         ADD ESP,4

00C919E1    EB 0D           JMP SHORT MyProjec.00C919F0

printf("您在E等级\n");

00C919E3    68 887BC900     PUSH MyProjec.00C97B88

00C919E8    E8 5EF6FFFF     CALL MyProjec.00C9104B

00C919ED    83C4 04         ADD ESP,4

system("pause");

00C919F0    8BF4            MOV ESI,ESP

00C919F2    68 987BC900     PUSH MyProjec.00C97B98             ; ASCII "pause"

00C919F7    FF15 68B1C900   CALL NEAR DWORD PTR DS:[C9B168]   ; ucrtbase.system

00C919FD    83C4 04         ADD ESP,4

练习

1、2月14号情人节,男孩为了表白自己心爱的女生,买了一束花,女生收到花后,询问男生花的价格,如果花的价格大于999元,女生回复同意交往;如果花的价格大于等于99元且小于999元,女生回复我们可以做朋友;如果花的价格大于50元且小于99元,女生回复我们改天再约吧!其它价格,女生则直接挂电话。题目要求:必须有题目分析步骤、流程图、请用C语言中if语句写出来。

2、请写出来下面的汇编对应的C语言代码。

00401010   push        ebp

00401011   mov         ebp,esp

00401013   sub         esp,44h

00401016   push        ebx

00401017   push        esi

00401018   push        edi

00401019   lea         edi,[ebp-44h]

0040101C   mov         ecx,11h

00401021   mov         eax,0CCCCCCCCh

00401026   rep stos    dword ptr [edi]

00401028   mov         dword ptr [ebp-4],0

0040102F   push  offset string

"\xc7\xeb\xca\xe4\xc8\xeb\xc4\xfa\xb5\xc4\xb3\xc9\xbc\xa8\xa3\xba" (004240

00401034   call        printf (00401290)

00401039   add         esp,4

0040103C   lea         eax,[ebp-4]

0040103F   push        eax

00401040   push        offset string "%d" (00424074)

00401045   call        scanf (00401230)

0040104A   add         esp,8

0040104D   cmp         dword ptr [ebp-4],5Ah

00401051   jg          main+4Fh (0040105f)

00401053   cmp         dword ptr [ebp-4],64h

00401057   jne         main+5Eh (0040106e)

00401059   cmp         dword ptr [ebp-4],64h

0040105D   jge         main+5Eh (0040106e)

0040105F  push       offset string "\xc4\xfa\xd4\xdaA\xb5\xc8\xbc\xb6\n" (00424064)

00401064   call        printf (00401290)

00401069   add         esp,4

0040106C   jmp         main+0CEh (004010de)

0040106E   cmp         dword ptr [ebp-4],50h

00401072   jg          main+70h (00401080)

00401074   cmp         dword ptr [ebp-4],5Ah

00401078   jne         main+7Fh (0040108f)

0040107A   cmp         dword ptr [ebp-4],5Ah

0040107E   jge         main+7Fh (0040108f)

00401080 push        offset string "\xc4\xfa\xd4\xdaB\xb5\xc8\xbc\xb6\n" (00424054)

00401085   call        printf (00401290)

0040108A   add         esp,4

0040108D   jmp         main+0CEh (004010de)

0040108F   cmp         dword ptr [ebp-4],46h

00401093   jg          main+91h (004010a1)

00401095   cmp         dword ptr [ebp-4],50h

00401099   jne         main+0A0h (004010b0)

0040109B   cmp         dword ptr [ebp-4],50h

0040109F   jge         main+0A0h (004010b0)

004010A1  push       offset string "\xc4\xfa\xd4\xdaC\xb5\xc8\xbc\xb6\n" (00424044)

004010A6   call        printf (00401290)

004010AB   add         esp,4

004010AE   jmp         main+0CEh (004010de)

004010B0   cmp         dword ptr [ebp-4],3Ch

004010B4   jg          main+0B2h (004010c2)

004010B6   cmp         dword ptr [ebp-4],46h

004010BA   jne         main+0C1h (004010d1)

004010BC   cmp         dword ptr [ebp-4],46h

004010C0   jge         main+0C1h (004010d1)

004010C2  push       offset string "\xc4\xfa\xd4\xdaD\xb5\xc8\xbc\xb6\n" (00424034)

004010C7   call        printf (00401290)

004010CC   add         esp,4

004010CF   jmp         main+0CEh (004010de)

004010D1  push       offset string "\xc4\xfa\xd4\xdaE\xb5\xc8\xbc\xb6\n" (00424024)

004010D6   call        printf (00401290)

004010DB   add         esp,4

004010DE   push        offset string "pause" (0042401c)

004010E3   call        system (00401120)

004010E8   add         esp,4

004010EB   xor         eax,eax

3、任意输入三个实数,将a、b、c三个数看作三个线段,判断a、b、c能否构成三角形。题目要求:必须有题目分析步骤、流程图、代码。额外要求:自己写的代码务必切到反汇编窗口,并理解每句代码对应的汇编。

4、编写一段程序,输入两个整数值,如何后者是前者的约数,则显示“B是A的约数”。

如果不是,则显示“B不是A的约数”。

5、判断输入的值是否为0。

6、判断输入的整数的个位数是否为5?

7、判断输入的整数是负数、正数还是0?

8、输入一个整数,并显示它的绝对值。

9、判断输入两个整数的大小。

本文摘自编程达人《汇编的角度——C语言》。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值