耐人尋味的 for...else...語句
老婆給當(dāng)程序員的老公打電話:“下班順路買一斤包子帶回來,如果看到賣西瓜的,買一個(gè)。”當(dāng)晚,程序員老公手捧一個(gè)包子進(jìn)了家門……老婆怒道:“你怎么就買了一個(gè)包子?”老公答曰:“因?yàn)榭吹搅速u西瓜的。”
程序員買西瓜的笑話可能大部分讀者都知道,今天寫的這篇文章和這個(gè)笑話有一定的關(guān)系。
任何編程語言都提供了 if…else… 語句,表示如果(if)滿足條件就做某件事,否則(else)就做另外一件事:
- if a==b:
- print("true")
- else:
- print("false")
然而,在 Python 中 else 不僅可以和 if 搭配使用,另一種特有的句法是 for…else …,它還可以和 while、try…except 組合使用,例如:
- for i in range(3):
- print(i)
- else:
- print("end")
- >>>
- 0
- 1
- 2
- end
但是,你會(huì)發(fā)現(xiàn) for…else… 與 if…else… 表現(xiàn)得不一樣,按照以往經(jīng)驗(yàn)來說,執(zhí)行了 for 語句塊的代碼就不執(zhí)行 else 里面了,反之亦然。
然而,我們看到的卻恰恰相反,for 循環(huán)結(jié)束之后接著又執(zhí)行了 else 語句塊,這就有點(diǎn)意思了,if … else … 翻譯成大白話就是 如果…否則…,而 for…else… 翻譯成白話成了 直到… 然后 …,為什么不把它寫成 for…then… 的句式呢?
for 循環(huán)遍歷空列表也會(huì)執(zhí)行 else 語句塊,因?yàn)樗钦M顺?for 循環(huán)的一種特例情況。
- for i in []:
- print(i)
- else:
- print("end")
- >>>
- 0
繼續(xù)探索,我們用 break 提前終止 for 循環(huán)
- for i in range(3):
- print(i)
- if i % 2 == 0:
- break
- else:
- print("end")
- >>>
- 0
循環(huán)遇到 break 退出后,整個(gè)語句就結(jié)束,else 語句塊也不執(zhí)行了。
綜上,我們可以得出這樣一個(gè)結(jié)論,只有當(dāng)循環(huán)里沒有遇到 break 時(shí),else 塊才會(huì)執(zhí)行。
Python 之父為什么要搞出這樣的一種語法糖出來呢?這是我們常人沒法理解的。不過「python之禪」告訴了我們答案: “Although that way may not be obvious at first unless you’re Dutch.”。
帶著這個(gè)問題,我也在 StackOver Flow 找了一下答案,在平時(shí)的開發(fā)中真的很少有 for…else… 的應(yīng)用場(chǎng)景,不過,像下面這種場(chǎng)景用 for else 還真是一種 pythonic 的用法。
當(dāng)你用 for 循環(huán)迭代查找列表的中的某個(gè)元素時(shí),如果找到了就提前退出,如果迭代完了還沒找到需要以另外一種形式通知調(diào)用者時(shí),用 for else 無疑是***的選擇。
- for i in mylist:
- if i == target:
- break
- process(i)
- else:
- raise ValueError("List argument missing terminal flag.")
如果不用 for…else… , 那么還需要專門建立一個(gè)臨時(shí)標(biāo)記變量來標(biāo)記是否已經(jīng)找到了
- found = False
- for i in mylist:
- if i == target:
- found = True
- break
- process(i)
- if not found:
- raise ValueError("List argument missing terminal flag.")
當(dāng)你想在房間里找某樣?xùn)|西時(shí),只要在任意位置找到了,就停止繼續(xù)搜查工作。但如果把整個(gè)房間都翻遍了,還沒找到我們想要的東西,需要告訴人家說:這兒沒有你要找的東西。遇到這樣的情況用 for … else ,除此之外,***不要用它。
【本文是51CTO專欄作者“劉志軍”的原創(chuàng)文章,作者微信公眾號(hào):Python之禪(VTtalk)】