Python語言技巧之三元運算符使用介紹
python不支持C/C++中的三元操作符 ?:,替代的方法是 ...if... else...
舉例,用下面的語法實現(xiàn)求三個數(shù)的最小值。
nD1 if nD1 < ( nD2 if nD2<nD3 else nD3) else (nD2 if nD2 < nD3 else nD3)
python三元運算符的正確方法
因為下周要用php寫項目,所以周末在家里重新看php的語法,看到三元描述符,突然想起來python是沒有三元描述符的,印象中依稀記得有模擬的實現(xiàn),于是上網(wǎng)上搜了一下。
(對應(yīng)C語言的 X ? V1:V2)
其中一種是:
(X and V1) or V2
正常情況下是不會有錯誤的,但是文章中也提到了,當(dāng)V1=""時,就會有問題
比如
print (True and '') or 'V'
print (False and '') or 'V'
輸出永遠都是: V
完美的解決方案是在《python核心編程中提到的》:
V1 if X else V2
原文如下:
如果你來自 C/C++ 或者是 Java 世界, 那么你很難忽略的一個事實就是 Python 在很長的一
段時間里沒有條件表達式(C ? X : Y), 或稱三元運算符. ( C 是條件表達式; X 是 C 為 True 時
的結(jié)果, Y 是 C 為 False 時的結(jié)果) 貴鐸·范·羅薩姆一直拒絕加入這樣的功能, 因為他認為應(yīng)
該保持代碼簡單, 讓程序員不輕易出錯. 不過在十年多后, 他放棄了, 主要是因為人們試著用
and 和 or 來模擬它, 但大多都是錯誤的. 根據(jù) FAQ , 正確的方法(并不唯一)是
(C and [X] or [Y])[0] . 唯一的問題是社區(qū)不同意這樣的語法. (你可以看一看 PEP 308, 其
中有不同的方案.) 對于Python 的這一問題,人們表達了極大的訴求.
貴鐸·范·羅薩姆最終選擇了一個最被看好(也是他最喜歡)的方案, 然后把它運用于標(biāo)準(zhǔn)庫中
的一些模塊. 根據(jù) PEP , "這個評審?fù)ㄟ^考察大量現(xiàn)實世界的案例, 包含不同的應(yīng)用, 以及由不同
程序員完成的代碼." 最后 Python 2.5 集成的語法確定為: X if C else Y .
如上文所說,該語法在python2.5才被加入,但是因為平時也不會用到2.4及以前的版本,所以也就夠用了~
現(xiàn)在大部分高級語言都支持“?”這個三元運算符(ternary operator),它對應(yīng)的表達式如下:condition ? value if true : value if false。很奇怪的是,這么常用的運算符python居然不支持!誠然,我們可以通過if-else語句表達,但是本來一行代碼可以完成的非要多行,明顯不夠簡潔。沒關(guān)系,在python里其實還是有對應(yīng)的表達方式的。
舉個例子:char *ret = (x!=0) ? "True" : "False"這行代碼對應(yīng)的python形式就是ret = (x and "True") or "False"(很簡單吧,事實上括號可以去掉)。運行時,python虛擬機會對賦值符右邊的布爾表達式(注意這里并非三元表達式)求值,返回值是最后一個被分析到的值。為什么是“最后一個被分析到的”而不是表達式中“最后一個”呢?因為布爾表達式有一個短路效應(yīng),比如a or b,如果a為真那么就不會分析b了。嗯,估計現(xiàn)在大家差不多明白了這行python代碼的原理了。如果x為真,由于字符串“True”也為真,于是返回"True",反之,x為假,那么就沒必要看字符串"True"了(短路效應(yīng)),直接返回"False"。
不難看出,三元運算在python中事實上可以通過借用布爾求值表達。然后,有時會有點小問題。舉個例子,char *ret = x ? "" or "VAL"。根據(jù)前面的例子,我們很自然想到在python里應(yīng)該這樣寫,ret = x and "" or "VAL"。錯了!不管x的布爾求值是真還是假,ret得到的總是"VAL"。奇怪么?不奇怪,因為在python中對空字符串的布爾求值為false,這樣x and ""永遠都是false,所以ret得到的自然總是"VAL"了。解決這個問題有兩種辦法,第一種,也是我喜歡的一種,就是寫成ret = not x and "VAL" or ""。第二種,麻煩一點ret=x and [""] or ["VAL"],然后每次取ret[0]作為返回值,這是因為[""]在布爾求值時值為true。
討論一:第一種方法代碼明顯要簡潔,效率也高,那么還有必要使用第二種么?當(dāng)然,第一種辦法有局限性,只有當(dāng)我們非常明確其中一個值布爾求值時不可能為false時才能使用。在我們的示例中,由于"VAL"肯定返回true所以可以使用。如果是兩個變量呢,像這樣ret=x and val1 or val2,你就只能老老實實寫成ret=x and [val1] or [val2],然后取ret[0]作為結(jié)果了。因為這行語句所表達的不是“當(dāng)x為真返回val1,否則返回val2”,而是“當(dāng)x為真并且val1為真返回val2,否則返回val2”。
討論二:大家都知道python里有l(wèi)ist和tuple,前面這行代碼ret=x and [""] or ["VAL"]我們就是通過list解決,有的人可能偏愛tuple,于是就會這樣寫ret=x and ("") or ("VAL")。錯了!這里ret[0]永遠都是空字符串(在2.5上測試)。這是我比較faint的一點,為啥[""]為真而("")為假呢?
最后,附上python對典型數(shù)值的布爾求值結(jié)果,這對我們書寫三元運算的等價語句很有用。
輸入 | 布爾求值 |
1,-1,[“”] | True |
0, “”, None, [], (), {}, (“”) | False |
相關(guān)文章
淺談pandas篩選出表中滿足另一個表所有條件的數(shù)據(jù)方法
今天小編就為大家分享一篇淺談pandas篩選出表中滿足另一個表所有條件的數(shù)據(jù)方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-02-02對YOLOv3模型調(diào)用時候的python接口詳解
今天小編就為大家分享一篇對YOLOv3模型調(diào)用時候的python接口詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-08-08Python實現(xiàn)將照片變成卡通圖片的方法【基于opencv】
這篇文章主要介紹了Python實現(xiàn)將照片變成卡通圖片的方法,涉及Python基于opencv庫進行圖片處理的相關(guān)操作技巧,需要的朋友可以參考下2018-01-01