python中for循环的用法_理解 Python 的 for 循环

在本篇博客中,我们将讨论 Python 中 for 循环的原理。

我们将从一组基本例子和它的语法开始,还将讨论与 for 循环关联的 else 代码块的用处。

然后我们将介绍迭代对象、迭代器和迭代器协议,还会学习如何创建自己的迭代对象和迭代器。

之后,我们将讨论如何使用迭代对象和迭代器实现 for 循环,以及利用 while 循环通过迭代器协议实现 for 循环逻辑。

最后,我们将反编译一个简单的 for 循环,并逐步介绍 Python 解释器在执行 for 循环时执行的指令,以满足大家的好奇心。这些有助于理解 for 循环运行时的内部工作原理。

Python的for循环

for 语句是 Python 中执行迭代的两个语句之一,另一个语句是 while。如果你对 Python 的迭代并不是很熟悉的话,Python中的迭代:for、while、break、以及continue语句是一个不错的切入点。

Python 中,for 循环用于遍历一个迭代对象的所有元素。循环内的语句段会针对迭代对象的每一个元素项目都执行一次。暂且可以将迭代对象想象成一个对象集合,我们可以一个个遍历里面的元素。我们将在下一节对迭代器迭代对象作详细说明。

一个简单的 for 循环

我们先从一个简单 for 循环开始,它遍历一个字符串列表并打印每一个字符串。

642c98ae3c6041f984fb77b73324ce0d.png

如你所见,这个循环实际上遍历了列表中的每一个单词并打印它们。也就是说,在循环的每一次遍历中,变量 word 都被指定为列表中的一个元素,然后执行 for 语句中的代码块。由于列表是一个有序的元素序列,所以循环也是以相同的顺序遍历这些元素。

带有 else 子句的 for 循环

Python 中的 for 循环可以选择是否关联一个 else 子句。else 子句中的代码块是在 for 循环完成后才开始执行的,即在迭代对象中的所有元素都遍历完毕之后。现在我们看一下如何扩展前面的示例以包含一个 else 条件(子句)。

4fb23cc1c7815d65aef4a1008d97473e.png

else 子句适用于何时?

你已经注意到,else 子句是在 for 循环完成之后才执行的。那么 else 代码块的意义是什么呢?for 循环之后的语句不是也是同样会执行吗?

我们很多时候会遇到这样一种情况,当满足某种条件时,中途结束 for 循环。且如果这个条件一直未满足,则希望执行另一组语句。我们通常使用布尔类型的标记实现,下面是一个例子。

e6f92704286fb19249b44088506f6c69.png

调用结果:

e6a3ded27030c5dc9099cc7cdf1ebd3f.png

而用 else 代码块的话,我们可以避免使用布尔类型的标记 found_item。我们看看如何使用 else 子句重写上面的方法。注意如果 for 循环中的 break 语句被触发执行,那么则会跳过 else 块。

6f926720851ecca4b0e03443a44a7d03.png

所以 else 代码块适用于 for 循环中有 break 语句的情况,且我们希望 break 条件没有被触发的时候执行一些语句。

否则,与 else 关联的语句只会在 for 循环结束时才执行。本文的最后一节查看反编译的字节码时你会看到这一点。

for 循环语法

我们已经看到了一些简单的例子,接下来以 for 循环的语法结束本节。

164248ed1cbf876df82b80b93e4b592a.png

基本上,对于 iterable 中的每一个元素,都会执行 set_of_statements_1。一旦所有的元素都迭代一遍,控制器将跳转到 else 代码块中执行 set_of_statements_2

注意,else 子句是可选的。如果没有发现 else 子句,循环会在所有元素都遍历完成后结束,并且控制器会转向程序之后的语句。

可迭代对象与迭代器

可迭代对象

在上一节,我们使用术语 iterable 来表示循环中被迭代的对象。现在我们来试着了解一下 Python 中的 iterable 对象是什么。

Python 中,一个 iterable 对象指在 for 循环中可以被迭代的任意对象。这意味着,当这个对象作为参数传递给 iter()方法时应该返回一个迭代器。我们来看一下 Python 中的一些常用的内置迭代的例子。

fa778872fbc09c5d764de120a01d615d.png

如你所见,当我们对一个 iterable 对象调用 iter() 时,它会返回一个迭代器对象。

迭代器

那么什么是迭代器呢?迭代器在 Python 中被定义为一个表现为流式数据的对象。基本上,如果我们将对象传递给内置的 next() 方法,它应该从与之关联的流式数据中返回下一个值。一旦所有的元素都遍历结束,它会抛出一个 *StopIteration* 异常。next() 方法的后续调用也都会抛出 *StopIteration* 异常。

我们用一个列表来试一下。

1f84ad9e647f87ef2b35c605c953cd9f.png

迭代器也是可迭代对象!但是...

有一个很有趣的事需要记一下,迭代器同样支持(强制要求支持迭代器协议) iter() 方法。这意味着我们可以对一个迭代器调用 iter() 方法并获取它自身的迭代器对象。

00f062f8389edfb30d73c774ccd30442.png

因此,我们可以在任何期望使用迭代器的地方使用它。比如,for 循环。

然而要注意一点,在像 list 这样的容器对象上调用 iter() 每次都会返回不同的迭代器,而在迭代器上调用 iter() 仅仅返回同一个迭代器。

9ac561c5449b3b7968984954647f03dc.png
410148bdb662fe1a752cf5060f47523a.png

所以如果你需要进行多次迭代,并且用迭代器替换普通容器或可迭代对象,那么第二次你会看到一个空的容器。

对一个列表迭代两次

请注意,这是按照我们的期望运行的。

462d4d41529df25da4c0fd74a8581702.png

对一个列表迭代器迭代两次

请注意,迭代器在第一次循环的时候就已经结束了,第二次我们看到的是一个空容器。

a6a2f6d91c32347ee15fdd3568515d0b.png

迭代器协议

前文我们看到了:

1. 一个可迭代对象,作为参数传递给 iter() 方法时返回一个迭代器

2. 一个迭代器,

1. 作为参数传递给 next() 方法时返回它的下一个元素或者在所有元素都遍历结束时抛 出 StopIteration 异常。

2. 作为参数传递给 iter() 方法时返回它自身。

迭代协议仅仅只是一种将对象定义为迭代器的标准方式。我们已经在前一节看到了这种协议的实际应用。根据协议,迭代器应该定义以下两个方法:

1. __next__()

1. 每次调用这个方法时,应该返回迭代器的下一个元素。一旦元素都遍历结束,它应该抛出 StopIteration 异常。

2. 当我们调动内置函数 next() 时,实际内部调用的是本方法。

2. __iter__()

1. 这个方法返回迭代器自身

2. 当我们调动内置函数 iter() 时,实际内部调用的是本方法。

自己写一个迭代器

现在我们已经知道迭代协议的原理,可以写一个自己的迭代器了。我们先看一个例子,下面我们创建了一个根据给定范围和步长的 Range 类。

8b76da188ad4c6b85259a0ce5fe95a6a.png

我们看一下它在 for 循环中是怎么工作的。

d1c89bc11863ad758c2fd9a7b13b4229.png

注意,Range 类的实例是迭代器也是可迭代对象。

自己写一个可迭代对象

我们还可以基于 Range 迭代器另外创建一个可迭代对象。它的作用是每当调用 __iter()__ 方法是返回一个新的迭代器,在这里,它应该返回一个新的 Range 对象。

b38ba10cf82ee12a4cfab01eaca302dd.png

在 for 循环中使用我们这个 RangeIterable。

cc7fd1733bf8ac1b4018f78d7250a522.png

for 循环工作原理

现在我们已经知道什么是迭代器和可迭代对象,接下来了解一下 for 循环是如何工作的。

再看一下前面的例子。

6282c8dd91de37a7027972acc0af1bd3.png

当我们执行上面的代码块时,发生了以下这些事情:

1. 在 for 语句内部对列表 ["You

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值