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


Posted in Python onJune 30, 2017

前言

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循环中。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

Python 相关文章推荐
Python实现从订阅源下载图片的方法
Mar 11 Python
Python使用装饰器进行django开发实例代码
Feb 06 Python
用Python分析3天破10亿的《我不是药神》到底神在哪?
Jul 12 Python
详解Python3中的迭代器和生成器及其区别
Oct 09 Python
对python判断ip是否可达的实例详解
Jan 31 Python
Python+PyQt5实现美剧爬虫可视工具的方法
Apr 25 Python
python tqdm 实现滚动条不上下滚动代码(保持一行内滚动)
Feb 19 Python
python matplotlib 绘图 和 dpi对应关系详解
Mar 14 Python
Python学习工具jupyter notebook安装及用法解析
Oct 23 Python
python3中for循环踩过的坑记录
Dec 14 Python
Python 使用SFTP和FTP实现对服务器的文件下载功能
Dec 17 Python
python中PyQuery库用法分享
Jan 15 Python
python 连接sqlite及简单操作
Jun 30 #Python
利用Python破解斗地主残局详解
Jun 30 #Python
Python实现的文本编辑器功能示例
Jun 30 #Python
Python构建XML树结构的方法示例
Jun 30 #Python
基于python的Tkinter编写登陆注册界面
Jun 30 #Python
Python使用微信SDK实现的微信支付功能示例
Jun 30 #Python
python实现的二叉树定义与遍历算法实例
Jun 30 #Python
You might like
一个显示某段时间内每个月的方法 返回由这些月份组成的数组
2012/05/16 PHP
jquery中的$(document).ready()使用小结
2014/02/14 Javascript
jquery 使用简明教程
2014/03/05 Javascript
js创建对象的区别示例介绍
2014/07/24 Javascript
C#中使用迭代器处理等待任务
2015/07/13 Javascript
Javascript连接Access数据库完整实例
2015/08/03 Javascript
jquery表单验证需要做些什么
2015/11/17 Javascript
关于cookie的初识和运用(js和jq)
2016/04/07 Javascript
Knockoutjs 学习系列(二)花式捆绑
2016/06/07 Javascript
easyui关于validatebox实现多重规则验证的方法(必看)
2017/04/12 Javascript
JS判断微信扫码的方法
2017/08/07 Javascript
weui框架实现上传、预览和删除图片功能代码
2017/08/24 Javascript
JS实现快速比较两个字符串中包含有相同数字的方法
2017/09/11 Javascript
iview的table组件自带的过滤器实现
2019/07/12 Javascript
[01:48]完美圣典齐天大圣至宝宣传片
2016/12/17 DOTA
Python标准库之多进程(multiprocessing包)介绍
2014/11/25 Python
Python实现简单的多任务mysql转xml的方法
2017/02/08 Python
python调用系统ffmpeg实现视频截图、http发送
2018/03/06 Python
详解如何管理多个Python版本和虚拟环境
2019/05/10 Python
Pycharm运行加载文本出现错误的解决方法
2019/06/27 Python
python数据预处理之数据标准化的几种处理方式
2019/07/17 Python
python连接、操作mongodb数据库的方法实例详解
2019/09/11 Python
Pytorch DataLoader 变长数据处理方式
2020/01/08 Python
python GUI库图形界面开发之PyQt5表单布局控件QFormLayout详细使用方法与实例
2020/03/06 Python
keras的ImageDataGenerator和flow()的用法说明
2020/07/03 Python
HTML5的一个显示电池状态的API简介
2015/06/18 HTML / CSS
Marlies Dekkers内衣美国官方网上商店:高端内衣品牌
2018/11/12 全球购物
中科方德软件测试面试题
2016/04/21 面试题
反腐倡廉警示教育活动总结
2014/05/05 职场文书
2014年残疾人工作总结
2014/12/06 职场文书
长城导游词300字
2015/01/30 职场文书
建党伟业的观后感
2015/06/01 职场文书
PHP 技巧 * SVG 保存为图片(分享图生成)
2021/04/02 PHP
Python天气语音播报小助手
2021/09/25 Python
MySQL数据库之存储过程 procedure
2022/06/16 MySQL
win7配置本地ftp服务器的图文教程
2022/08/05 Servers