新书推荐:7.3 for语句

本节必须掌握的知识点:

        示例二十四

        代码分析

        汇编解析

7.3.1 示例二十四

for语句语法形式:

for(表达式1;表达式2;表达式3)

{

语句块;

}

●语法解析:

第一步:执行表达式1,表达式1初始化循环变量;

第二步:执行表达式2,表达式2条件判断语句,如果表达式条件为真,则执行循环体内的语句块,否则退出for语句;

第三步:执行循环体内的语句块;

第四步:执行表达式3后置表达式;

第五步:继续执行下一轮循环,直到表达式2的条件为假,跳出for循环语句。

执行顺序:表达式1->表达式2->循环语句块->表达式3->表达式2->循环语句块->表达式3->表达式2->循环语句块……一直循环下去,直到到表达式2为假时,停止循环。接下来我们示例二十四来理解for语句的执行流程。

示例代码二十四

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

分析需求:构建一个for循环语句,当i<= 5时,重复执行for语句内的重复块。

设计程序结构框架:循环结构for语句。

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

       int i;//定义一个int类型的整型局部变量i,并将其初始化为0,作为循环变量。

  ●第三步:分析算法。

       利用for语句,当i<= 5时,重复执行for语句内的重复块。先判断后执行。

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

 int main(void) {

        定义一个int类型整型循环变量i;

        表达式1:将循环变量i初始化为0;

         表达式2:当i <= 5时执行大括号内的循环语句块

        表达式3:执行完printf语句后,循环变量i加1

        for (i = 0; i <= 5; i++){

            调用printf函数打印信息"i = %d\n"                                     

        }

        调用printf函数打印信息"i = %d\n" 

        system("pause");

        return 0;                                                                                                    

}

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

图7-3 示例二十四for循环语句

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

/*

   for语句

*/

#include <stdio.h>

#include <stdlib.h>

int main(void) {

    int i;

    for (i = 0; i <= 5; i++)

    {

        printf("i = %d\n", i);

    }

        printf("i = %d\n", i);

    system("pause");                                        

    return 0;                          

}    

●输出结果:

i = 0

i = 1

i = 2

i = 3

i = 4

i = 5

i = 6

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

参见反汇编代码。

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

(无)

7.3.2 代码分析

分析for语句执行流程:

第一步:执行 int i = 0;

第二步:执行 i<=5;0<5条件为真,继续往下执行 ;

第三步:执行printf(“i = %d”,i);输出 i = 0;

第四步:执行i++; i=0+1,i=1;

第五步:执行i<=5;1<5条件为真,继续往下执行;

第六步:执行printf(“i = %d”,i);输出 i =1;

第七步:执行i++;i=1+1,i=2;

第八步:执行i<=5;2<5条件为真,继续往下执行;

第九步:执行printf(“i = %d”,i);输出 i =2;

第十步:执行i++;i=2+1,i=3;

第十一步:执行i<=5;3<5条件为真,继续往下执行;

第十二步:执行printf(“i = %d”,i);输出 i =3;

第十三步:执行i++;i=1+3,i=4;

第十四步:执行i<=5;4<5条件为真,继续往下执行;

第十五步:执行printf(“i = %d”,i);输出 i =4;

第十六步:执行i++;i=1+4,i=5;

第十七步:执行i<=5;5=5条件为真,继续往下执行;

第十八步:执行printf(“i = %d”,i);输出 i =5;

第十九步:执行i++;i=1+5,i=6;

第二十步:执行i<=5;6>5条件为假,退出循环;

退出for循环后,再次输出i的值为6。

这是这个程序的正向分析流程,我们看下它的反汇编是怎么体现的。

7.3.3 汇编解析

汇编代码

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

include vcIO.inc

.data

i sdword  ?

.const    

szMsg db "i = %d",0dh,0ah,0

.code     

start:

       mov i,0

NEXT:           

       .while i <= 5

              invoke printf,offset szMsg,i 

              inc sdword ptr i;

              jmp NEXT           

       .endw           

       ;

       invoke printf,offset szMsg,i

       ;     

       invoke _getch

       ret                       

end start

●输出结果:

i = 0

i = 1

i = 2

i = 3

i = 4

i = 5

i = 6

       上述汇编代码使用了.while高级汇编伪指令(没有.for高级汇编伪指令),实现了C语言中的for循环语句。

 

结论

从本质上看,C语言中的for语句就是while语句的简化形式。“mov i,0”语句对应表达式1;条件表达式“i <= 5”对应表达式2;“inc sdword ptr i”语句对应表达式3。

反汇编代码

    int i;

    for (i = 0; i <= 5; i++)

00391838  mov         dword ptr [i],0  ;表达式1:初始化循环变量i=0

0039183F  jmp         main+3Ah (039184Ah);无条件跳转到表达式2语句

00391841  mov         eax,dword ptr [i] 

00391844  add         eax,1              ;表达式3:变量i加一

00391847  mov         dword ptr [i],eax 

0039184A  cmp         dword ptr [i],5  ;表达式2:比较变量i和5的大小

    int i;

    for (i = 0; i <= 5; i++)

0039184E  jg          main+53h (0391863h) ;如果i>5,则跳出for循环,否则执行循环语句

    {

        printf("i = %d\n", i)

00391850  mov         eax,dword ptr [i] 

00391853  push        eax                              

00391854  push        offset string "i = %d\n" (0397B30h) ;循环语句

00391859  call        _printf (039104Bh) 

0039185E  add         esp,8 

    }

00391861  jmp         main+31h (0391841h) ;无条件跳转到表达式3

    printf("i = %d\n", i);

00391863  mov         eax,dword ptr [i]  ;for循环外的下一条语句

00391866  push        eax 

00391867  push        offset string "i = %d\n" (0397B30h) 

0039186C  call        _printf (039104Bh) 

00391871  add         esp,8 

    system("pause");

00391874  mov         esi,esp 

00391876  push        offset string "pause" (0397B38h) 

0039187B  call        dword ptr [__imp__system (039B168h)] 

00391881  add         esp,4 

 

总结

在for语句中:

表达式1基本上是用来定义循环控制变量或者说对循环语句的开始进行控制;

表达式2一般是用来判断条件是否满足的语句,满足条件则继续执行,不满足则退出for语句;

表达式3称为步长,一般是用来控制循环次数的,;

语句块:可以做任何事情。

实验四十八: for语句表现形式1

在VS中新建项目7-3-2.c:

/*

   for语句的表现形式1:省略第一个表达式

*/

#include <stdio.h>

#include <stdlib.h>

int main(void) {

    int i = 0;         // 在for循环之前初始化

    for (; i < 10; i++)// for语句中省去第一个表达式。但是分号不能少

    {

        printf("■");

    }

    printf("\n");

    system("pause");

    return 0;

}

●输出结果:

■■■■■■■■■■

请按任意键继续. . .

       上述代码中的表达式1放置在for语句之前,for语句内只保留了一个’;’,其他保持不变。接下来我们通过反汇编代码观察一下这种形式的for语句的执行流程。                     

■反汇编代码:

    int i = 0;         //  在for循环之前初始化

009B1838  mov         dword ptr [i],0  ;表达式1前置

    for (; i < 10; i++)// for语句中省去第一个表达式。但是分号不能少

009B183F  jmp         main+3Ah (09B184Ah

009B1841  mov         eax,dword ptr [i] 

009B1844  add         eax,1                ;表达式3

009B1847  mov         dword ptr [i],eax 

009B184A  cmp         dword ptr [i],0Ah  ;表达式2

    for (; i < 10; i++)// for语句中省去第一个表达式。但是分号不能少

009B184E  jge         main+4Fh (09B185Fh;当i>=10,退出循环

    {

        printf("■");

009B1850  push        offset string "\xa1\xf6" (09B7B30h) 

009B1855  call        _printf (09B104Bh) 

009B185A  add         esp,4 

    }

009B185D  jmp         main+31h (09B1841h;跳转到表达式3

    printf("\n");

009B185F  push        offset string "\n" (09B80D8h) 

009B1864  call        _printf (09B104Bh) 

009B1869  add         esp,4   

 

结论

由上述反汇编代码可知,表达式1前置,对应for语句的执行流程没有任何改变。

练习

  1. 请读者书写程序7-3-2.c伪代码,并绘制流程图。
  2. 请读者将7-3-2.c翻译成汇编语言实现。
  3. 请读者分析7-3-2.c的反汇编代码。

实验四十九: for语句表现形式2

在VS中新建项目7-3-3.c:

/*

   for语句的表现形式2:省略第二个表达式

*/

#include <stdio.h>

#include <stdlib.h>

int main(void) {

    for (int i = 0;; i++)//变量i的作用域仅限for语句块内

    {

        if (i > 10)

        {

            break;//如果i>10则跳出循环,break是跳出for循环

        }

        printf("■");

    }

    //printf("i = %d\n",i);

    system("pause");

    return 0;

}

       ●输出结果:

■■■■■■■■■■■请按任意键继续. . .

       实验四十九的代码需要注意两点:

       1.变量i的定义放在了for语句内,那么变量i的作用域仅限for语句块。如果将下面的语句“printf("i = %d\n",i);”屏蔽去掉,编译时会提示错误信息“error C2065: “i”: 未声明的标识符”。在C语言中,变量是区分作用域的,也就是有效范围。如果是全局变量,则保存在全局区,在整个程序范围内都是有效的。如果局部变量定义在函数内,则保存在函数堆栈中,仅在函数内有效,函数运行结束后释放堆栈空间,局部变量也就消失了。此外还有两种类型的变量static静态变量和STL线程存储变量,这里暂不做介绍。

       在C语言中,不同作用域的变量名可以相同,因为作用域不同,编译器编译时也会自动区分。

       2.for语句的表达式2是一个条件表达式,其作用是控制循环结束条件。实验四十九中的代码将表达式2放置在循环语句块内,使用if语句判断条件,当i>10时,执行break语句,退出循环。break语句的作用就是退出循环,我们将在本章7.5节继续介绍break语句的使用方法。

【注】for语句中的表达式2分号不可以缺省。

       接下来我们再看一下反汇编代码:

    for (int i = 0;; i++)//变量i的作用域仅限for语句块内

00B01838  mov         dword ptr [ebp-8],0  ;表达式1,[ebp-8]为局部变量i

    for (int i = 0;; i++)//变量i的作用域仅限for语句块内

00B0183F  jmp         main+3Ah (0B0184Ah

00B01841  mov         eax,dword ptr [ebp-8]  ;表达式3

00B01844  add         eax,1 

00B01847  mov         dword ptr [ebp-8],eax 

    {

        if (i > 10);表达式2

00B0184A  cmp         dword ptr [ebp-8],0Ah 

00B0184E  jle         main+42h (0B01852h) 

        {

            break;//如果i>10则跳出循环,break是跳出for循环

00B01850  jmp         main+51h (0B01861h)  ;退出循环

        }

        printf("■");循环语句

00B01852  push        offset string "\xa1\xf6" (0B07B30h) 

00B01857  call        _printf (0B0104Bh) 

00B0185C  add         esp,4 

    }

00B0185F  jmp         main+31h (0B01841h)  ;跳转到表达式3

    //printf("i = %d\n",i);

    system("pause");

00B01861  mov         esi,esp 

       上述反汇编代码中一个明显的变化就是变量i使用[ebp-8]来代替。为了进一步验证变量i的作用域,我们再做一个实验。

实验五十: for语句循环变量作用域

在VS中新建项目7-3-4.c:

/*

   for语句循环变量作用域

*/

#include <stdio.h>

#include <stdlib.h>

int main(void) {

    int i = 20;//作用域main函数内有效

    for (int i = 0;; i++)//变量i的作用域仅限for语句块内

    {

        printf("\ni = %d", i);

        if (i > 10)

        {

            break;//如果i>10则跳出循环,break是跳出for循环

        }

        printf("■");

    }

    printf("\ni = %d\n",i);//i=20

    system("pause");

    return 0;

}

       ●输出结果:

i = 0■

i = 1■

i = 2■

i = 3■

i = 4■

i = 5■

i = 6■

i = 7■

i = 8■

i = 9■

i = 10■

i = 11

i = 20

请按任意键继续. . .

上述代码中存在两个变量i,编译时并没有报错,说明变量名符号没有冲突。再看打印信息,for语句内打印的变量i从0~11。最后一个打印输出的变量i的值为20,显然是第一个变量i的值。

请读者查看一下反汇编代码,第一个变量i使用“[i]”表示,而第二个变量i使用“[ebp-14h]表示”。其余内容和使用四十九相同,不再赘述。

实验五十一: for语句形式3

在VS中新建项目7-3-5.c:

/*

   for语句形式3:缺省表达式3

*/

#include <stdio.h>

#include <stdlib.h>

int main(void) {

    for (int i = 0; i < 10;)

    {

        printf("■");

        i++;

    }

    system("pause");

    return 0;

}

●输出结果:

■■■■■■■■■■请按任意键继续. . .

上述代码将表达式3放置在循环语句块的结尾处,这应该和我们预想的一样,之前for语句的反汇编代码的执行流程就是如此。接下来我们再看一下反汇编代码的执行流程。

●反汇编代码:

    for (int i = 0; i < 10;)

002D1838  mov         dword ptr [ebp-8],0  ;表达式1

    for (int i = 0; i < 10;)

002D183F  cmp         dword ptr [ebp-8],0Ah  ;表达式2

002D1843  jge         main+4Dh (02D185Dh)  ;i>=10跳出循环

    {

        printf("■");循环语句

002D1845  push        offset string "\xa1\xf6" (02D7B30h) 

002D184A  call        _printf (02D104Bh) 

002D184F  add         esp,4 

        i++;表达式3

002D1852  mov         eax,dword ptr [ebp-8] 

002D1855  add         eax,1 

002D1858  mov         dword ptr [ebp-8],eax 

    }

002D185B  jmp         main+2Fh (02D183Fh;继续循环

    system("pause");

002D185D  mov         esi,esp

上述反汇编代码和之前示例中的反汇编代码执行流程稍有不同,表达式3按照正常的代码编写顺序放置在printf语句之后。其实之前的反汇编代码也是按照代码的编写顺序进行编译的。

【注意】for (int i = 0; i < 10;)语句内的最后一个表达式;可以缺省。

实验五十二: for语句形式4

在VS中新建项目7-3-6.c:

/*

   for语句形式4:缺省全部3个表达式

*/

#include <stdio.h>

#include <stdlib.h>

int main(void) {

    for (;;)

    {

        printf("■");

    }

    system("pause");

    return 0;

}

●输出结果:

不停的打印■,陷入死循环。

为什么会是这样呢?因为当表达式全部置空时,相当于没有初始值,没有控制语句,没有后置表达式,那么它将永远的循环下去,直到内存溢出,这就是所谓的死循环。

   ●反汇编代码:

    for (;;)

    {

        printf("■");

011E1838  push        offset string "\xa1\xf6" (011E7B30h) 

    for (;;)

    {

        printf("■");

011E183D  call        _printf (011E104Bh) 

011E1842  add         esp,4 

    }

011E1845  jmp         main+28h (011E1838h;跳到for语句块起始位置

    system("pause");

011E1847  mov         esi,esp 

       上述反汇编代码可以看到,程序中没有循环控制变量,没有循环结束条件,因此陷入了无限循环。我们应绝对避免此类情况的发生。

练习

1、递增显示从0到输入的正整数为止的各个整数(for语句)。

2、输入一个整数,连续显示出该整数个*(for语句)。

3、输入规定数量个整数并显示出它们的合计值和平均值。

4、编写一个程序,求1到n的和。n的值从键盘输入。

5、根据输入的整数,循环显示1234567890,显示的位数和输入的整数值相同。

请输入一个整数:25

1234567890123456789012345

6、显示输入的整数值的所有约数。

7、编写一段程序,输入一个整数值,显示该数值以下的所有奇数。

整数值:15

1 3 5 7 9 11 13 15

8、编写一段程序,输入一个整数值,显示该整数值个‘*'。每显示5个就进行换行。

显示多少个*:12

*****

*****

** 

*

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/841963.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【错题集-编程题】kotori 和迷宫(BFS / DFS)

牛客对应题目链接&#xff1a;kotori和迷宫 (nowcoder.com) 一、分析题目 迷宫问题的扩展。 二、代码 #include <iostream> #include <cstring> #include <queue>using namespace std;const int N 35; int x1, y1; // 标记起点位置 int n, m; char arr[N][…

电机控制系列模块解析(23)—— 同步机初始位置辨识

一、两个常见问题 为什么感应电机&#xff08;异步机&#xff09;不需要初始位置辨识&#xff1f;&#xff08;因此感应电机转子磁场在定子侧进行励磁&#xff0c;其初始位置可以始终人为定义为0&#xff09; 为什么同步磁阻电机需要初始位置辨识&#xff1f;&#xff08;因为…

五分钟搭建一个Suno AI音乐站点

五分钟搭建一个Suno AI音乐站点 在这个数字化时代&#xff0c;人工智能技术正以惊人的速度改变着我们的生活方式和创造方式。音乐作为一种最直接、最感性的艺术形式&#xff0c;自然也成为了人工智能技术的应用场景之一。今天&#xff0c;我们将以Vue和Node.js为基础&#xff…

【软件设计师】计算机组成原理

1、数据的表示 1.1 进制转换 整型有4种进制形式&#xff1a; 1.十进制&#xff08;D&#xff09;&#xff1a; 都是以0-9这九个数字组成&#xff0c;不能以0开头。 2.二进制&#xff08;B&#xff09;&#xff1a; 由0和1两个数字组成。 3.八进制&#xff08;O&#xff09;&am…

嵌入式进阶——数码管

&#x1f3ac; 秋野酱&#xff1a;《个人主页》 &#x1f525; 个人专栏:《Java专栏》《Python专栏》 ⛺️心若有所向往,何惧道阻且长 文章目录 数码管结构移位寄存器原理图移位寄存器数据流程移位寄存器控制流程移位寄存器串联实现数码管显示 数码管结构 共阴与共阳 共阳数码…

linux centos循环ping网段ip

循环ping&#xff0c;检测ip是否可用&#xff0c;ping通为正在使用yes&#xff0c;no为不通 vim test.sh#!/bin/bash ip"192.168.1."echo "ping log:" > ./ping.txt for i in {1..128} doping -c 1 -w 1 -W 1 $ip$i | grep -q "ttl" &&a…

Android跨进程通信--Binder机制及AIDL是什么?

文章目录 Binder机制Binder是什么&#xff1f;Binder相对于其他几种跨进程通信方式&#xff0c;有什么区别&#xff1f;谈一下 Binder IPC 通信过程&#xff1a;具体的通讯过程是什么&#xff1f;Binder如何处理发送请求与接收请求?Binder是通过什么方式来进行内存映射的&…

linux入门到精通-第十九章-libevent(开源高性能事件通知库)

目录 参考什么是libevent应用核心实现libevent的地基event_base等待事件产生&#xff0c;循环监听event_loop退出循环监听event_base_loopexit创建事件工作流程 安装一&#xff08;源码安装&#xff0c;推荐&#xff09;现在源码配置编译安装验证安装 安装二&#xff08;可能因…

KingbaseES数据库merge语法

数据库版本&#xff1a;KingbaseES V008R006C008B0014 简介 MERGE 语句是一种用于数据操作的 SQL 语句&#xff0c;它能够根据指定的条件将 INSERT、UPDATE 和 DELETE 操作结合到单个语句中。其主要作用是在目标表和源表之间进行数据比较和同步&#xff0c;根据条件的匹配情况来…

RAG概述(二):Advanced RAG 高级RAG

目录 概述 Advanced RAG Pre-Retrieval预检索 优化索引 增强数据粒度 粗粒度 细粒度 展开说说 优化索引 Chunk策略 Small2Big方法 元数据 引入假设性问题 对齐优化 混合检索 查询优化 查询扩展 查询转换 Post-Retrieval后检索 参考 概述 Native RAG&#…

springboot vue 开源 会员收银系统 (4) 门店模块开发

前言 完整版演示 前面我们对会员系统 springboot vue 开源 会员收银系统 (3) 会员管理的开发 实现了简单的会员添加 下面我们将从会员模块进行延伸 门店模块的开发 首先我们先分析一下常见门店的管理模式 常见的管理形式为总公司 - 区域管理&#xff08;若干个门店&#xff…

C语言 | Leetcode C语言题解之第113题路径总和II

题目&#xff1a; 题解&#xff1a; int** ret; int retSize; int* retColSize;int* path; int pathSize;typedef struct {struct TreeNode* key;struct TreeNode* val;UT_hash_handle hh; } hashTable;hashTable* parent;void insertHashTable(struct TreeNode* x, struct Tr…

C++干货 --类和对象(二)

前言&#xff1a; 上文中&#xff0c;我们介绍了类这一重要知识点&#xff0c;包括为什么要有类、类的使用方法、封装、以及对象实例化。详情可以去看我的文章&#xff1a;写文章-CSDN创作中心C干货 --类和对象(一)-CSDN博客写文章-CSDN创作中心 这篇文章&#xff0c;我们简单…

我学会了用插件来辅助PostgreSQL,可拷,很刑!

作者&#xff1a;IT邦德 中国DBA联盟(ACDU)成员&#xff0c;10余年DBA工作经验&#xff0c; Oracle、PostgreSQL ACE CSDN博客专家及B站知名UP主&#xff0c;全网粉丝10万 擅长主流Oracle、MySQL、PG、高斯及Greenplum备份恢复&#xff0c; 安装迁移&#xff0c;性能优化、故障…

【蓝桥杯省赛真题44】python计算N+N的值 中小学青少年组蓝桥杯比赛 算法思维python编程省赛真题解析

目录 python计算NN的值 一、题目要求 1、编程实现 2、输入输出 二、算法分析 三、程序编写 四、程序说明 五、运行结果 六、考点分析 七、 推荐资料 1、蓝桥杯比赛 2、考级资料 3、其它资料 python计算NN的值 第十四届蓝桥杯青少年组python省赛真题 一、题目要求…

【教学类-58-05】黑白三角拼图05(2-10宫格,每个宫格随机1张-6张,带空格纸,1页3张黑白3张白卡)

背景需求&#xff1a; 【教学类-58-04】黑白三角拼图04&#xff08;2-10宫格&#xff0c;每个宫格随机1张-6张&#xff0c;带空格纸&#xff0c;1页6张黑白&#xff0c;1张6张白卡&#xff09;-CSDN博客文章浏览阅读582次&#xff0c;点赞16次&#xff0c;收藏3次。【教学类-58…

Mac安装pytorch

先下载 Anaconda | The Operating System for AI 网速慢&#xff0c;用中国大陆镜像&#xff1a;NJU Mirror 之前装python3时用的是pip3&#xff0c;这里说一下这pip与conda的区别 Conda和pip都是Python包管理工具&#xff0c;用于安装和管理Python包 包管理范围&#xff1a…

汽车数据应用构想(一)

自从电动汽车GB/T32960标准颁布&#xff0c;要求所有电动汽车必须上传数据开始&#xff0c;各车厂就开始花费大量的人力物力&#xff0c;用于数据的上传与存储。同时随着智能化、网联化的趋势&#xff0c;不断丰富上传数据的内容与数量。数据已成为车厂的重要资产&#xff0c;但…

微服务:eureka的搭建,以及服务注册、服务发现、负载均衡

eureka 搭建 新建一个Module,maven项目&#xff0c;导入依赖。 <dependencies><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-netflix-eureka-server</artifactId></dependency><…

24年湖南教资认定即将开始,别被照片卡审!

24年湖南教资认定即将开始&#xff0c;别被照片卡审&#xff01;