Python天坑系列(一):while 1比while True更快?

转载 2015年04月20日 16:25:23


0. 前言

前些天被Python的多线程坑了一把,因此产生了写一个《Python天坑系列》博客的想法,说说我碰到的那些Python的坑。

而天坑这个词呢,一方面指Python的坑,另一方面也说明本系列文章也是个坑,对于会写什么内容、有多少篇、多久更新一次、什么时间更新我都无法确定,哈哈(看,之前已经3个月没有更新过了!)。

本篇是系列的第一篇,讲的内容是Python的bool类型。

1. 前提

1.1 bool是int的子类

根据PEP285中Review部分第6条所述,bool类是从int类继承而来的,这样可以极大的简化实现(C代码中调用PyInt_Check()的地方仍将继续工作)。

1.2 Python2中True/False不是关键字,但Python3中是

我们可以导入keyword模块,来查看关键字:

而在Python3中,关键字中添加了True/False/None。

由于Python2中True/False不是关键字,因此我们可以对其进行任意的赋值:

2. True + True = 2

由于bool是继承自int的子类,因此为了保证向下兼容性,在进行算术运算中,True/False会被当作int值来执行。

3. While 1比While True快?

首先来看一个比较while 1和while True循环的脚本,两个函数中,除了1和True的区别之外,其他地方完全相同。

执行结果:
while one: 1.37000703812
while_true: 2.07638716698

可以看出wihle 1的执行时间约为while True的2/3。

那么,这是为什么呢?

其实这就是前提中提到的关键字的问题。由于Python2中,True/False不是关键字,因此我们可以对其进行任意的赋值,这就导致程序在每次循环时都需要对True/False的值进行检查;而对于1,则被程序进行了优化,而后不会再进行检查。

我们可以通过dis模块来查看while_one和while_true的字节码,下面的程序是对刚才的程序进行了一定的简化后的版本。

执行的结果是:

可以看出,正如上面所讲到的,在while True的时候,字节码中多出了几行语句,正是这几行语句进行了True值的检查。

而在Python3中,由于True/False已经是关键字了,不允许进行重新赋值,因此,其执行结果与while 1不再有区别(好吧,我这没有Python3的环境,就不去验证了,网上有人验证过了)。但是由于Python2的使用十分广泛,因此大家不得不注意这个可能会降低性能的地方。

4. if x == True: 还是 if x:

在PEP285中,还提到了这两种写法的比较。PEP285中认为,==具有传递性,a==b, b==c会被化简为a==c。也就是说,如果选择前一种写法的话,6和7在if语句中都应该被认为是真值,那么就会造成6==True==7,被化简为6==7的问题,因此后一种写法才是正确的。

现在,让我们偏个题,假设x就是True,那么程序的执行效率又如何呢?

执行结果是:
if_x_eq_true: 0.212558031082
if_x: 0.144327878952

让我们再来看看字节码(程序未作修改,dis的使用方式同上,因此不再给出程序):

可以清晰的看到第9行比第14行,多出了检查True值和进行比较的操作。

也就是说,不论从遵循PEP的规范,还是执行效率,或者程序的简洁性来说,我们都应该使用if x:,而不是if x == True:来进行比较。同理,那些if x is not None:之类的语句也应当被简化为if x:(如果要比较的是非值,而不必须是None的话)。

5. References

http://legacy.python.org/dev/peps/pep-0285/
http://stackoverflow.com/questions/3815359/while-1-vs-for-whiletrue-why-is-there-a-difference

本文内容遵从CC3.0版权协议,转载请注明:转自Pythoner

本文链接地址:Python天坑系列(一):while 1比while True更快?

Python学习笔记(While循环)

和其他编程语言一样,Python同样提供并支持循环语句。循环语句允许我们执行一条或多条语句多次。 Python中提供的循环语句有for循环和while循环.while循环是指在给定的条件成立时(tr...
  • jun_life
  • jun_life
  • 2016年06月18日 17:06
  • 12168

python学习——while True的用法

在学习过程中,经常能遇到采用while True的用法。下面以一个例子进行说明:建立一个用户登录系统,用户输入用户名和密码,如果正确就可以进入系统。1、我自己最开始的写法:d = {} ...
  • geerniya
  • geerniya
  • 2017年08月24日 09:56
  • 1618

用python实现从1加到100的三种方法: for循环,while循环,导入模块法

第一种是for循环 第二种是while循环 第三种,导入模块的内建函数reduce
  • fang_zz
  • fang_zz
  • 2016年05月29日 10:41
  • 24718

Python中while和for的区别

在c、c++、Java、C#等语言中,while和for的用法是基本一致的,但是在Python中,这两个循环还是存在一定的差别的 当遍历序列或数组时, 只能用for,while用来遍历会出现死循环...
  • u012135425
  • u012135425
  • 2016年09月24日 20:25
  • 1977

Python的while循环

 i = 3;while i < 10: print i; i = i + 1;else: print "end of while"; 
  • xxkkff
  • xxkkff
  • 2009年07月14日 19:56
  • 19301

python 的 do ~ while 语法

本文摘至: http://ama-ch.hatenablog.com/entry/20080425/1209110237 Python不支持do〜while语法、while(无限循环)和break组...
  • robertsong2004
  • robertsong2004
  • 2014年12月03日 15:28
  • 18947

python 学习笔记-while 循环

  • 2017年11月02日 01:03
  • 1.23MB
  • 下载

python中的while语句

循环使用 else 语句 在 python 中,while … else 在循环条件为 false 时执行 else 语句块: 实例 #!/usr/bin/python ...
  • LL1234566
  • LL1234566
  • 2017年09月28日 10:22
  • 101

Python for, while循环后面加else的作用

Python中的for, while循环后面紧接着是可以加else的,这种设计有什么好处呢?         一般情况下,for, while循环如果在计数器用尽跳出时,用不用else语句对结果...
  • u012083681
  • u012083681
  • 2013年12月18日 22:31
  • 4053

python的while和for循环

while语句,提供了编写通用循环的一种方法,而for语句是用来遍历序列对象内的元素,并对每个元素运行一个代码块。break,continue用在循环内,跳出整个循环或者跳出一次循环。 一、while...
  • u010229420
  • u010229420
  • 2015年02月22日 11:31
  • 3397
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Python天坑系列(一):while 1比while True更快?
举报原因:
原因补充:

(最多只允许输入30个字)