## 第06课:循环结构 ### 应用场景 我们在写程序的时候,一定会遇到需要重复执行某条指令或某些指令的场景。例如用程序控制机器人踢足球,如果机器人持球而且还没有进入射门范围,那么我们就要一直发出让机器人向球门方向移动的指令。在这个场景中,让机器人向球门方向移动就是一个需要重复的动作,当然这里还会用到上一课讲的分支结构来判断机器人是否持球以及是否进入射门范围。再举一个简单的例子,如果要实现每隔1秒中在屏幕上打印一次“hello, world”并持续打印一个小时,我们肯定不能够直接把`print('hello, world')`这句代码写3600遍,这里我们需要构造循环结构。 所谓循环结构,就是程序中控制某条或某些指令重复执行的结构。在Python中构造循环结构有两种做法,一种是`for-in`循环,另一种是`while`循环。 ### for-in循环 如果明确的知道循环执行的次数,我们推荐使用`for-in`循环,例如输出100行的”hello, world“。 被`for-in`循环控制的语句块也是通过缩进的方式来构造的,这一点跟分支结构完全相同,大家看看下面的代码就明白了。 ```Python """ 用for循环实现1~100求和 Version: 0.1 Author: 骆昊 """ total = 0 for x in range(1, 101): total += x print(total) ``` 需要说明的是上面代码中的`range(1, 101)`可以用来构造一个从`1`到`100`的范围,当我们把这样一个范围放到`for-in`循环中,就可以通过前面的循环变量`x`依次取出从`1`到`100`的整数。当然,`range`的用法非常灵活,下面给出了一个例子: - `range(101)`:可以用来产生0到100范围的整数,需要注意的是取不到101。 - `range(1, 101)`:可以用来产生1到100范围的整数,相当于前面是闭区间后面是开区间。 - `range(1, 101, 2)`:可以用来产生1到100的奇数,其中2是步长,即每次递增的值。 - `range(100, 0, -2)`:可以用来产生100到1的偶数,其中-2是步长,即每次递减的值。 知道了这一点,我们可以用下面的代码来实现1~100之间的偶数求和。 ```Python """ 用for循环实现1~100之间的偶数求和 Version: 0.1 Author: 骆昊 """ total = 0 for x in range(2, 101, 2): total += x print(total) ``` ### while循环 如果要构造不知道具体循环次数的循环结构,我们推荐使用`while`循环。`while`循环通过一个能够产生`bool`值的表达式来控制循环,当表达式的值为`True`时则继续循环,当表达式的值为`False`时则结束循环。 下面我们通过一个“猜数字”的小游戏来看看如何使用`while`循环。猜数字游戏的规则是:计算机出一个`1`到`100`之间的随机数,玩家输入自己猜的数字,计算机给出对应的提示信息(大一点、小一点或猜对了),如果玩家猜中了数字,计算机提示用户一共猜了多少次,游戏结束,否则游戏继续。 ```Python """ 猜数字游戏 Version: 0.1 Author: 骆昊 """ import random # 产生一个1-100范围的随机数 answer = random.randint(1, 100) counter = 0 while True: counter += 1 number = int(input('请输入: ')) if number < answer: print('大一点') elif number > answer: print('小一点') else: print('恭喜你猜对了!') break # 当退出while循环的时候显示用户一共猜了多少次 print(f'你总共猜了{counter}次') ``` ### break和continue 上面的代码中使用`while True`构造了一个条件恒成立的循环,也就意味着如果不做特殊处理,循环是不会结束的,这也就是常说的“死循环”。为了在用户猜中数字时能够退出循环结构,我们使用了`break`关键字,它的作用是提前结束循环。需要注意的是,`break`只能终止它所在的那个循环,这一点在使用嵌套循环结构时需要引起注意,下面的例子我们会讲到什么是嵌套的循环结构。除了`break`之外,还有另一个关键字是`continue`,它可以用来放弃本次循环后续的代码直接让循环进入下一轮。 ### 嵌套的循环结构 和分支结构一样,循环结构也是可以嵌套的,也就是说在循环中还可以构造循环结构。下面的例子演示了如何通过嵌套的循环来输出一个乘法口诀表(九九表)。 ```Python """ 打印乘法口诀表 Version: 0.1 Author: 骆昊 """ for i in range(1, 10): for j in range(1, i + 1): print(f'{i}*{j}={i * j}', end='\t') print() ``` 很显然,在上面的代码中,外层循环用来控制一共会产生`9`行的输出,而内层循环用来控制每一行会输出多少列。内层循环中的输出就是九九表一行中的所有列,所以在内层循环完成时,有一个`print()`来实现换行输出的效果。 ### 循环的例子 #### 例子1:输入一个正整数判断它是不是素数。 > **提示**:素数指的是只能被1和自身整除的大于1的整数。 ```Python """ 输入一个正整数判断它是不是素数 Version: 0.1 Author: 骆昊 """ num = int(input('请输入一个正整数: ')) end = int(num ** 0.5) is_prime = True for x in range(2, end + 1): if num % x == 0: is_prime = False break if is_prime and num != 1: print(f'{num}是素数') else: print(f'{num}不是素数') ``` #### 例子2:输入两个正整数,计算它们的最大公约数和最小公倍数。 > **提示**:两个数的最大公约数是两个数的公共因子中最大的那个数;两个数的最小公倍数则是能够同时被两个数整除的最小的那个数。 ```Python """ 输入两个正整数计算它们的最大公约数和最小公倍数 Version: 0.1 Author: 骆昊 """ x = int(input('x = ')) y = int(input('y = ')) for factor in range(x, 0, -1): if x % factor == 0 and y % factor == 0: print(f'{x}和{y}的最大公约数是{factor}') print(f'{x}和{y}的最小公倍数是{x * y // factor}') break ``` ### 简单的总结 学会了Python中的分支结构和循环结构,我们就可以解决很多实际的问题了。通过这节课的学习,大家应该已经知道了可以用`for`和`while`关键字来构造循环结构。**如果知道循环的次数,我们通常使用**`for`**循环**;如果**循环次数不能确定,可以用**`while`**循环**。在循环中还**可以使用**`break`**来提前结束循环**。