摘要:循環(huán)本質(zhì)是不斷地調(diào)用迭代器的方法,直到有異常為止,所以任何可迭代對(duì)象都可以作用在循環(huán)中。
for 循環(huán)用于迭代容器對(duì)象中的元素,這些對(duì)象可以是列表、元組、字典、集合、文件,甚至可以是自定義類或者函數(shù),例如:
作用于列表
>>> 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
可能有人不經(jīng)要問(wèn),為什么這么多不同類型對(duì)象都支持 for 語(yǔ)句,還有哪些類型的對(duì)象可以作用在 for 語(yǔ)句中呢?回答這個(gè)問(wèn)題之前,我們先要了解 for 循環(huán)背后的執(zhí)行原理。
for 循環(huán)是對(duì)容器進(jìn)行迭代的過(guò)程,什么是迭代?迭代就是從某個(gè)容器對(duì)象中逐個(gè)地讀取元素,直到容器中沒有更多元素為止。那么,哪些對(duì)象支持迭代操作?任何對(duì)象都可以嗎?先隨便自定義一個(gè)類試試,看行不行:
>>> class MyRange: ... def __init__(self, num): ... self.num = num ... >>> for i in MyRange(10): ... print(i) ... Traceback (most recent call last): File "", line 1, in TypeError: "MyRange" object is not iterable
錯(cuò)誤堆棧日志非常清楚地告訴我們,MyRange 不是一個(gè)可迭代對(duì)象,所以它不能用于迭代,那么到底什么樣的對(duì)象才稱得上是可迭代對(duì)象(iterable)呢?
可迭代對(duì)象需要實(shí)現(xiàn)__iter__方法,并返回一個(gè)迭代器,什么是迭代器呢?迭代器只需要實(shí)現(xiàn) __next__方法。現(xiàn)在我們就來(lái)驗(yàn)證一下列表為什么支持迭代:
>>> x = [1,2,3] >>> its = x.__iter__() # x有此方法,說(shuō)明列表是可迭代對(duì)象 >>> its>>> its.__next__() # its有此方法,說(shuō)明its是迭代器 1 >>> its.__next__() 2 >>> its.__next__() 3 >>> its.__next__() Traceback (most recent call last): File " ", line 1, in StopIteration
從試驗(yàn)結(jié)果來(lái)看,列表是一個(gè)可迭代對(duì)象,因?yàn)樗鼘?shí)現(xiàn)了 __iter__方法,并且返回了一個(gè)迭代器對(duì)象(list_iterator),因?yàn)樗鼘?shí)現(xiàn)了 __next__方法。我們看到它不斷地調(diào)用__next__方法,其實(shí)就是不斷地迭代獲取容器中的元素,直到容器中沒有更多元素拋出 StopIteration 異常為止。
那么 for 語(yǔ)句又是如何循環(huán)的呢?到這里,恐怕你也猜到了,它的步驟是:
先判斷對(duì)象是否為可迭代對(duì)象,不是的話直接報(bào)錯(cuò),拋出TypeError異常,是的話,調(diào)用 __iter__方法,返回一個(gè)迭代器
不斷地調(diào)用迭代器的__next__方法,每次按序返回迭代器中的一個(gè)值
迭代到最后,沒有更多元素了,就拋出異常 StopIteration,這個(gè)異常 python 自己會(huì)處理,不會(huì)暴露給開發(fā)者
對(duì)于元組,字典,字符串也是同樣的道理,弄明白了 for 的執(zhí)行原理之后,我們就可以實(shí)現(xiàn)自己的迭代器用在 for 循環(huán)中。
前面的 MyRange 報(bào)錯(cuò)是因?yàn)樗鼪]有實(shí)現(xiàn)迭代器協(xié)議里面的這兩個(gè)方法,現(xiàn)在繼續(xù)改進(jìn):
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: # 達(dá)到某個(gè)條件時(shí)必須拋出此異常,否則會(huì)無(wú)止境地迭代下去 raise StopIteration()
因?yàn)樗鼘?shí)現(xiàn)了__next__方法,所以 MyRange 本身已經(jīng)是一個(gè)迭代器了,所以 __iter__返回的就是對(duì)象本身 self。現(xiàn)在用在 for 循環(huán)中試試:
for i in MyRange(3): print(i) # 輸出 0 1 2
有沒有發(fā)現(xiàn),自定義的 MyRange 功能和內(nèi)建函數(shù) range很相似。for 循環(huán)本質(zhì)是不斷地調(diào)用迭代器的__next__方法,直到有 StopIteration 異常為止,所以任何可迭代對(duì)象都可以作用在for循環(huán)中。
原文首發(fā)于公眾號(hào):python之禪
博客:https://foofish.net/how-for-w...
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/45548.html
摘要:第二次迭代時(shí),使用,那么,就是強(qiáng)行修改表達(dá)式的值為,本來(lái)是的,那么與都有返回值,它們的返回值是當(dāng)前迭代遇到時(shí),后面表達(dá)式的值,其實(shí)就是當(dāng)前迭代中后面的參數(shù)。 yield 為了精通 yield ,你必須要理解:當(dāng)你調(diào)用這個(gè)函數(shù)的時(shí)候,函數(shù)內(nèi)部的代碼并不立馬執(zhí)行 ,這個(gè)函數(shù)只是返回一個(gè)生成器對(duì)象,這有點(diǎn)蹊蹺不是嗎。 那么,函數(shù)內(nèi)的代碼什么時(shí)候執(zhí)行呢?當(dāng)你使用for進(jìn)行迭代的時(shí)候.現(xiàn)在到了關(guān)...
摘要:我們把對(duì)象傳給內(nèi)置的方法,會(huì)返回一個(gè)迭代器,循環(huán)就是使用這個(gè)模式來(lái)實(shí)現(xiàn)適用于所有的對(duì)象。舉例當(dāng)?shù)髟贈(zèng)]有元素可以迭代時(shí)會(huì)引發(fā)一個(gè)異常。因此,對(duì)于你提的例子我們使用迭代器來(lái)重新定義一下。 如果你從迭代層面來(lái)理解的話可能對(duì)于for的工作原理會(huì)有更深的理解。首先我們來(lái)使用dir查看一下對(duì)于range、str這兩個(gè)的不一樣的類型有什么共同點(diǎn)。 >>> dir(range) [__class_...
摘要:針對(duì)尾遞歸優(yōu)化的語(yǔ)言可以通過(guò)尾遞歸防止棧溢出。尾遞歸事實(shí)上和循環(huán)是等價(jià)的,沒有循環(huán)語(yǔ)句的編程語(yǔ)言只能通過(guò)尾遞歸實(shí)現(xiàn)循環(huán)。標(biāo)準(zhǔn)的解釋器沒有針對(duì)尾遞歸做優(yōu)化,任何遞歸函數(shù)都存在棧溢出的問(wèn)題。 python 頭部: #!/usr/bin/env python # -*- coding: utf-8 -*- 函數(shù)的參數(shù) Python的函數(shù)具有非常靈活的參數(shù)形態(tài),既可以實(shí)現(xiàn)簡(jiǎn)單的調(diào)用,又可以傳入...
摘要:本文重點(diǎn)掌握可迭代的對(duì)象的定義掌握可迭代對(duì)象迭代器與生成器之間的關(guān)系和異同熟悉標(biāo)準(zhǔn)庫(kù)中生成器。二迭代器迭代器介紹迭代器用于從集合中取出元素的對(duì)象。若想再次迭代須重建迭代器。迭代器檢查方式調(diào)用,。區(qū)別可迭代的對(duì)象不是迭代器。 導(dǎo)語(yǔ):本文章記錄了本人在學(xué)習(xí)Python基礎(chǔ)之控制流程篇的重點(diǎn)知識(shí)及個(gè)人心得,打算入門Python的朋友們可以來(lái)一起學(xué)習(xí)并交流。 本文重點(diǎn): 1、掌握可迭代的對(duì)象的...
閱讀 2585·2021-09-26 10:13
閱讀 5999·2021-09-08 10:46
閱讀 696·2019-08-30 15:53
閱讀 2970·2019-08-29 16:13
閱讀 2763·2019-08-26 12:23
閱讀 3490·2019-08-26 11:24
閱讀 1097·2019-08-23 18:09
閱讀 1036·2019-08-23 17:08