详解Python中for循环是如何工作的

作者:FOOFISH 时间:2021-07-04 17:41:40 

前言

for...in 是Python程序员使用最多的语句,for 循环用于迭代容器对象中的元素,这些对象可以是列表、元组、字典、集合、文件,甚至可以是自定义类或者函数,例如:

作用于列表


>>> for elem in [1,2,3]:
...  print(elem)
...
1
2
3

作用于元组


>>> for i in ("zhang", "san", 30):
...  print(i)
...
zhang
san
30

作用于字符串


>>> for c in "abc":
...  print(c)
...
a
b
c

作用于集合


>>> for i in {"a","b","c"}:
...  print(i)
...
b
a
c

作用于字典


>>> for k in {"age":10, "name":"wang"}:
...  print(k)
...
age
name

作用于文件


>>> for line in open("requirement.txt"):
...  print(line, end="")
...
Fabric==1.12.0
Markdown==2.6.7

可能有人不经要问,为什么这么多不同类型对象都支持 for 语句,还有哪些类型的对象可以作用在 for 语句中呢?回答这个问题之前,我们先要了解 for 循环背后的执行原理。

for 循环是对容器进行迭代的过程,什么是迭代?迭代就是从某个容器对象中逐个地读取元素,直到容器中没有更多元素为止。那么,哪些对象支持迭代操作?任何对象都可以吗?先随便自定义一个类试试,看行不行:


>>> class MyRange:
...  def __init__(self, num):
...   self.num = num
...
>>> for i in MyRange(10):
...  print(i)
...
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: 'MyRange' object is not iterable

错误堆栈日志非常清楚地告诉我们,MyRange 不是一个可迭代对象,所以它不能用于迭代,那么到底什么样的对象才称得上是可迭代对象(iterable)呢?

可迭代对象需要实现__iter__方法,并返回一个迭代器,什么是迭代器呢?迭代器只需要实现 __next__方法。现在我们就来验证一下列表为什么支持迭代:


>>> x = [1,2,3]
>>> its = x.__iter__() # x有此方法,说明列表是可迭代对象
>>> its
<list_iterator object at 0x100f32198>

>>> its.__next__() # its有此方法,说明its是迭代器
1
>>> its.__next__()
2
>>> its.__next__()
3
>>> its.__next__()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
StopIteration

从试验结果来看,列表是一个可迭代对象,因为它实现了 __iter__方法,并且返回了一个迭代器对象(list_iterator),因为它实现了 __next__方法。我们看到它不断地调用__next__方法,其实就是不断地迭代获取容器中的元素,直到容器中没有更多元素抛出 StopIteration 异常为止。

那么 for 语句又是如何循环的呢?到这里,恐怕你也猜到了,它的步骤是:

  • 先判断对象是否为可迭代对象,不是的话直接报错,抛出TypeError异常,是的话,调用 __iter__方法,返回一个迭代器

  • 不断地调用迭代器的__next__方法,每次按序返回迭代器中的一个值

  • 迭代到最后,没有更多元素了,就抛出异常 StopIteration,这个异常 python 自己会处理,不会暴露给开发者

详解Python中for循环是如何工作的

对于元组,字典,字符串也是同样的道理,弄明白了 for 的执行原理之后,我们就可以实现自己的迭代器用在 for 循环中。

前面的 MyRange 报错是因为它没有实现迭代器协议里面的这两个方法,现在继续改进:


class MyRange:
def __init__(self, num):
 self.i = 0
 self.num = num

def __iter__(self):
 return self

def __next__(self):
 if self.i < self.num:
  i = self.i
  self.i += 1
  return i
 else:
  # 达到某个条件时必须抛出此异常,否则会无止境地迭代下去
  raise StopIteration()

因为它实现了__next__方法,所以 MyRange 本身已经是一个迭代器了,所以 __iter__返回的就是对象本身 self。现在用在 for 循环中试试:


for i in MyRange(3):
print(i)
# 输出
0
1
2

有没有发现,自定义的 MyRange 功能和内建函数 range很相似。for 循环本质是不断地调用迭代器的__next__方法,直到有 StopIteration 异常为止,所以任何可迭代对象都可以作用在for循环中。

来源:https://foofish.net/how-for-works-in-python.html

标签:python,for循环
0
投稿

猜你喜欢

  • Windows11使用Cpython 编译文件报错 error: Unable to find vcvarsall.bat 完美解决方法

    2021-03-14 19:03:19
  • Django博客系统注册之创建用户模块应用

    2021-08-06 15:15:20
  • 一文详解Golang中net/http包的实现原理

    2024-05-28 15:23:08
  • Python爬虫HTPP请求方法有哪些

    2023-07-25 16:55:06
  • Python运行提示缺少模块问题解决方案

    2023-06-24 02:16:23
  • python正则表达式实现自动化编程

    2022-01-08 12:24:33
  • 自己重新写了一个JavaScript的对象克隆函数

    2008-08-03 16:47:00
  • Python3 安装PyQt5及exe打包图文教程

    2021-09-24 12:43:56
  • oracle学习笔记(三)

    2012-01-05 19:28:42
  • vue实现Input输入框模糊查询方法

    2024-04-28 09:20:46
  • 用CSS实现柱状图(Bar Graph)的方法(四)—table实现复杂柱状图

    2008-05-28 12:55:00
  • python爬虫之Appium爬取手机App数据及模拟用户手势

    2023-12-28 00:10:46
  • 使用Python手工计算x的算数平方根,来自中国古人的数学智慧

    2021-12-07 01:29:53
  • 5招优化MySQL插入方法

    2009-04-02 10:49:00
  • 简单了解Python多态与属性运行原理

    2021-03-13 21:29:42
  • python实现合并两个数组的方法

    2022-05-30 21:40:11
  • 解决mysql安装时出现error Nr.1045问题的方法

    2024-01-18 11:34:30
  • MySQL之select、distinct、limit的使用

    2024-01-22 04:41:20
  • 如何通过Python的pyttsx3库将文字转为音频

    2023-01-11 19:54:59
  • Django ValuesQuerySet转json方式

    2021-12-05 07:15:34
  • asp之家 网络编程 m.aspxhome.com