python不支援C/C++中的三元操作符 ?:,替代的方法是 ...if... else...
舉例,用下面的文法實現求三個數的最小值。
nD1 if nD1 < ( nD2 if nD2<nD3 else nD3) else (nD2 if nD2 < nD3 else nD3)
python三元運算子的正確方法
因為下周要用php寫項目,所以周末在家裡重新看php的文法,看到三元描述符,突然想起來python是沒有三元描述符的,印象中依稀記得有類比的實現,於是上網上搜了一下。
(對應C語言的 X ? V1:V2)
其中一種是:
(X and V1) or V2
正常情況下是不會有錯誤的,但是文章中也提到了,當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 時
的結果, Y 是 C 為 False 時的結果) 貴鐸·范·羅薩姆一直拒絕加入這樣的功能, 因為他認為應
該保持代碼簡單, 讓程式員不輕易出錯. 不過在十年多後, 他放棄了, 主要是因為人們試著用
and 和 or 來類比它, 但大多都是錯誤的. 根據 FAQ , 正確的方法(並不唯一)是
(C and [X] or [Y])[0] . 唯一的問題是社區不同意這樣的文法. (你可以看一看 PEP 308, 其
中有不同的方案.) 對於Python 的這一問題,人們表達了極大的訴求.
貴鐸·范·羅薩姆最終選擇了一個最被看好(也是他最喜歡)的方案, 然後把它運用於標準庫中
的一些模組. 根據 PEP , "這個評審通過考察大量現實世界的案例, 包含不同的應用, 以及由不同
程式員完成的代碼." 最後 Python 2.5 整合的文法確定為: X if C else Y .
如上文所說,該文法在python2.5才被加入,但是因為平時也不會用到2.4及以前的版本,所以也就夠用了~
現在大部分進階語言都支援“?”這個三元運算子(ternary operator),它對應的運算式如下:condition ? value if true : value if false。很奇怪的是,這麼常用的運算子python居然不支援!誠然,我們可以通過if-else語句表達,但是本來一行代碼可以完成的非要多行,明顯不夠簡潔。沒關係,在python裡其實還是有對應的表達方式的。
舉個例子:char *ret = (x!=0) ? "True" : "False"這行代碼對應的python形式就是ret = (x and "True") or "False"(很簡單吧,事實上括弧可以去掉)。運行時,python虛擬機器會對賦值符右邊的布林運算式(注意這裡並非三元運算式)求值,傳回值是最後一個被分析到的值。為什麼是“最後一個被分析到的”而不是運算式中“最後一個”呢?因為布林運算式有一個短路效應,比如a or b,如果a為真那麼就不會分析b了。嗯,估計現在大家差不多明白了這行python代碼的原理了。如果x為真,由於字串“True”也為真,於是返回"True",反之,x為假,那麼就沒必要看字串"True"了(短路效應),直接返回"False"。
不難看出,三元運算在python中事實上可以通過借用布爾求值表達。然後,有時會有點小問題。舉個例子,char *ret = x ? "" or "VAL"。根據前面的例子,我們很自然想到在python裡應該這樣寫,ret = x and "" or "VAL"。錯了!不管x的布爾求值是真還是假,ret得到的總是"VAL"。奇怪麼?不奇怪,因為在python中對Null 字元串的布爾求值為false,這樣x and ""永遠都是false,所以ret得到的自然總是"VAL"了。解決這個問題有兩種辦法,第一種,也是我喜歡的一種,就是寫成ret = not x and "VAL" or ""。第二種,麻煩一點ret=x and [""] or ["VAL"],然後每次取ret[0]作為傳回值,這是因為[""]在布爾求值時值為true。
討論一:第一種方法代碼明顯要簡潔,效率也高,那麼還有必要使用第二種嗎?當然,第一種辦法有局限性,只有當我們非常明確其中一個值布爾求值時不可能為false時才能使用。在我們的樣本中,由於"VAL"肯定返回true所以可以使用。如果是兩個變數呢,像這樣ret=x and val1 or val2,你就只能老老實實寫成ret=x and [val1] or [val2],然後取ret[0]作為結果了。因為這行語句所表達的不是“當x為真返回val1,否則返回val2”,而是“當x為真並且val1為真返回val2,否則返回val2”。
討論二:大家都知道python裡有list和tuple,前面這行代碼ret=x and [""] or ["VAL"]我們就是通過list解決,有的人可能偏愛tuple,於是就會這樣寫ret=x and ("") or ("VAL")。錯了!這裡ret[0]永遠都是Null 字元串(在2.5上測試)。這是我比較faint的一點,為啥[""]為真而("")為假呢?
最後,附上python對典型數值的布爾求值結果,這對我們書寫三元運算的等價語句很有用。
輸入 |
布爾求值 |
1,-1,[“”] |
True |
0, “”, None, [], (), {}, (“”) |
False |