海象運(yùn)算符是在 PEP 572 被提出的,直到 3.8 版本合入發(fā)布。
它的英文原名叫 Assignment Expressions
,翻譯過(guò)來(lái)也就是 賦值表達(dá)式
,不過(guò)現(xiàn)在大家更普遍地稱之為海象運(yùn)算符,就是因?yàn)樗L(zhǎng)得真的太像海象了。
1. 第一個(gè)用法:if/else
可能有朋友是第一次接觸這個(gè)新特性,所以還是簡(jiǎn)單的介紹一下這個(gè)海象運(yùn)算符有什么用?
在 Golang 中的條件語(yǔ)句可以直接在 if 中運(yùn)算變量的獲取后直接對(duì)這個(gè)變量進(jìn)行判斷,可以讓你少寫一行代碼
import "fmt" func main() { if age := 20;age > 18 { fmt.Println("已經(jīng)成年了") } }
若在 Python 3.8 之前,Python 必須得這樣子寫
age = 20 if age > 18: print("已經(jīng)成年了")
但有了海象運(yùn)算符之后,你可以和 Golang 一樣(如果你沒學(xué)過(guò) Golang,那這里要注意,Golang 中的 :=
叫短變量聲明,意思是聲明并初始化,它和 Python 中的 :=
不是一個(gè)概念)
if (age:= 20) > 18: print("已經(jīng)成年了")
2. 第二個(gè)用法:while
在不使用 海象運(yùn)算符之前,使用 while 循環(huán)來(lái)讀取文件的時(shí)候,你也許會(huì)這么寫
file = open("demo.txt", "r") while True: line = file.readline() if not line: break print(line.strip())
但有了海象運(yùn)算符之后,你可以這樣
file = open("demo.txt", "r") while (line := file.readline()): print(line.strip())
使用它替換以往的無(wú)限 while 循環(huán)寫法更為驚艷
比如,實(shí)現(xiàn)一個(gè)需要命令行交互輸入密碼并檢驗(yàn)的代碼,你也許會(huì)這樣子寫
while True: p = input("Enter the password: ") if p == "youpassword": break
有了海象運(yùn)算符之后,這樣子寫更為舒服
while (p := input("Enter the password: ")) != "youpassword": continue
3. 第三個(gè)用法:推導(dǎo)式
這個(gè)系列的文章,幾乎每篇都能看到推導(dǎo)式的身影,這一篇依舊如此。
在編碼過(guò)程中,我很喜歡使用推導(dǎo)式,在簡(jiǎn)單的應(yīng)用場(chǎng)景下,它簡(jiǎn)潔且不失高效。
如下這段代碼中,我會(huì)使用列表推導(dǎo)式得出所有會(huì)員中過(guò)于肥胖的人的 bmi 指數(shù)
members = [ {"name": "小五", "age": 23, "height": 1.75, "weight": 72}, {"name": "小李", "age": 17, "height": 1.72, "weight": 63}, {"name": "小陳", "age": 20, "height": 1.78, "weight": 82}, ] count = 0 def get_bmi(info): global count count += 1 print(f"執(zhí)行了 {count} 次") height = info["height"] weight = info["weight"] return weight / (height**2) # 查出所有會(huì)員中過(guò)于肥胖的人的 bmi 指數(shù) fat_bmis = [get_bmi(m) for m in members if get_bmi(m) > 24] print(fat_bmis)
輸出如下
執(zhí)行了 1 次 執(zhí)行了 2 次 執(zhí)行了 3 次 執(zhí)行了 4 次 [25.88057063502083]
可以看到,會(huì)員數(shù)只有 3 個(gè),但是 get_bmi 函數(shù)卻執(zhí)行了 4 次,原因是在判斷時(shí)執(zhí)行了 3 次,而在構(gòu)造新的列表時(shí)又重復(fù)執(zhí)行了一遍。
如果所有會(huì)員都是過(guò)于肥胖的,那最終將執(zhí)行 6 次,這種在大量的數(shù)據(jù)下是比較浪費(fèi)性能的,因此對(duì)于這種結(jié)構(gòu),我通常會(huì)使用傳統(tǒng)的for 循環(huán) + if 判斷。
fat_bmis = [] # 查出所有會(huì)員中過(guò)于肥胖的人的 bmi 指數(shù) for m in members: bmi = get_bmi(m) if bmi > 24: fat_bmis.append(bmi)
在有了海象運(yùn)算符之后,你就可以不用在這種場(chǎng)景下做出妥協(xié)。
# 查出所有會(huì)員中過(guò)于肥胖的人的 bmi 指數(shù) fat_bmis = [bmi for m in members if (bmi := get_bmi(m)) > 24]
最終從輸出結(jié)果可以看出,只執(zhí)行了 3 次
執(zhí)行了 1 次 執(zhí)行了 2 次 執(zhí)行了 3 次 [25.88057063502083]
這里僅介紹了列表推導(dǎo)式,但在字典推導(dǎo)式和集合推導(dǎo)式中同樣適用。不再演示。
海象運(yùn)算符,是一個(gè)新奇的特性,有不少人覺得這樣這種特性會(huì)破壞代碼的可讀性。確實(shí)在一個(gè)新鮮事物剛出來(lái)時(shí)是會(huì)這樣,但我相信經(jīng)過(guò)時(shí)間的沉淀后,越來(lái)越多的人使用它并享受它帶來(lái)的便利時(shí),這種爭(zhēng)議也會(huì)慢慢消失在歷史的長(zhǎng)河中。