Python语言技巧之三元运算符使用介绍


Posted in Python onMarch 04, 2013

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中对空字符串的布尔求值为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的一点,为啥[""]为真而("")为假呢?

    最后,附上python对典型数值的布尔求值结果,这对我们书写三元运算的等价语句很有用。


输入 布尔求值
1,-1,[“”] True
0, “”, None, [], (), {}, (“”) False
Python 相关文章推荐
python网络编程之TCP通信实例和socketserver框架使用例子
Apr 25 Python
python面向对象_详谈类的继承与方法的重载
Jun 07 Python
python判断字符串是否是json格式方法分享
Nov 07 Python
python实现从文件中读取数据并绘制成 x y 轴图形的方法
Oct 14 Python
Python设计模式之建造者模式实例详解
Jan 17 Python
Python Flask框架扩展操作示例
May 03 Python
Python3内置模块random随机方法小结
Jul 13 Python
python 输出列表元素实例(以空格/逗号为分隔符)
Dec 25 Python
关于pytorch处理类别不平衡的问题
Dec 31 Python
Python面向对象封装操作案例详解 II
Jan 02 Python
用Python的绘图库(matplotlib)绘制小波能量谱
Apr 17 Python
python基于tkinter制作m3u8视频下载工具
Apr 24 Python
python解决字典中的值是列表问题的方法
Mar 04 #Python
python实现的各种排序算法代码
Mar 04 #Python
python 获取本机ip地址的两个方法
Feb 25 #Python
把大数据数字口语化(python与js)两种实现
Feb 21 #Python
python正则表达式修复网站文章字体不统一的解决方法
Feb 21 #Python
Python操作Mysql实例代码教程在线版(查询手册)
Feb 18 #Python
python的常见命令注入威胁
Feb 18 #Python
You might like
PHP基于CURL进行POST数据上传实例
2014/11/10 PHP
php开发时容易忘记的一些技术细节
2016/02/03 PHP
thinkPHP5实现的查询数据库并返回json数据实例
2017/10/23 PHP
PHP格式化显示时间date()函数代码
2018/10/03 PHP
如何在PHP中使用数组
2020/06/09 PHP
JavaScript入门教程(7) History历史对象
2009/01/31 Javascript
JavaScript的模块化:封装(闭包),继承(原型) 介绍
2013/07/22 Javascript
jquery插件开发之实现jquery手风琴功能分享
2014/03/10 Javascript
javascript实现2048游戏示例
2014/05/04 Javascript
js获取日期:昨天今天和明天、后天
2014/06/11 Javascript
JS获取客户端IP地址、MAC和主机名的7个方法汇总
2014/07/21 Javascript
Javascript基础教程之函数对象和属性
2015/01/18 Javascript
JS+CSS实现另类带提示效果的竖向导航菜单
2015/10/15 Javascript
Node.js开发者必须了解的4个JS要点
2016/02/21 Javascript
深入解析Vue 组件命名那些事
2017/07/18 Javascript
分享Bootstrap简单表格、表单、登录页面
2017/08/04 Javascript
如何理解Vue的render函数的具体用法
2017/08/30 Javascript
npm scripts 使用指南详解
2018/10/08 Javascript
javascript中call,apply,bind的区别详解
2020/12/11 Javascript
Python程序员鲜为人知但你应该知道的17个问题
2014/06/04 Python
Python for Informatics 第11章之正则表达式(四)
2016/04/21 Python
Python实现按中文排序的方法示例
2018/04/25 Python
Python对象与引用的介绍
2019/01/24 Python
浅谈pyqt5中信号与槽的认识
2019/02/17 Python
python从zip中删除指定后缀文件(推荐)
2019/12/05 Python
Python 随机生成测试数据的模块:faker基本使用方法详解
2020/04/09 Python
Scrapy中如何向Spider传入参数的方法实现
2020/09/28 Python
HTML5新增加标签和功能概述
2016/09/05 HTML / CSS
德国户外装备、登山运动和攀岩商店:tapir store
2020/02/12 全球购物
北京某科技有限公司C# .net笔试题
2014/09/27 面试题
2014年秋季开学典礼致辞
2014/08/02 职场文书
民政局个人整改措施
2014/09/24 职场文书
比赛主持人开场白
2015/05/29 职场文书
幼儿园体操比赛口号
2015/12/25 职场文书
解决pytorch-gpu 安装失败的记录
2021/05/24 Python
联想win10摄像头打不开怎么办?win10笔记本摄像头打不开解决办法
2022/04/08 数码科技