python中實現問號表達式


現在大部分高級語言都支持“?”這個三元運算符(ternary operator),它對應的表達式如下:condition ? value if true : value if false。很奇怪的是,這么常用的運算符python居然不支持!誠然,我們可以通過if-else語句表達,但是本來一行代碼可以完成的非要多行,明顯不夠簡潔。沒關系,在python里其實還是有對應的表達方式的。 

 

(1) variable = a if exper else b
(2) variable = (exper and [b] or [c])[0]
(3) variable = exper and b or c

 

上面三種用法都可以達到目的,類似C語言中 variable = exper ? b : c;即:如果exper表達式的值為true則variable = b,否則,variable = c

例如:

a, b = 1, 2
max = a if a > b else b
max2 = (a > b and [a] or [b])[0]  # list
max3 = a > b and a or b
print max,max2,max3

 

 

舉例分析第3種方法

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中對空字符串的布爾求值為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]永遠都是空字符串(在2.5上測試)。這是我比較faint的一點,為啥[""]為真而("")為假呢? 

 

參考:http://www.jb51.net/article/41974.htm


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM