编写同时兼容Python2.x与Python3.x版本的代码的几个示例


Posted in Python onMarch 30, 2015

编写兼容Python2.x与3.x代码

当我们正处于Python 2.x到Python 3.x的过渡期时,你可能想过是否可以在不修改任何代码的前提下能同时运行在Python 2和3中。这看起来还真是一个合理的诉求,但如何开始呢?哪些Python 2 代码在 3.x 解释器执行时容易出状况呢?
print vs print()

如果你想的和我一样,你或许会说print语句,这是个很好的着手点,先简单展示一下,print在2.x中是一条语句,而在3.x中它是一个关键字或者是保留字。换句话说,因为这个变化涉及到语言的语法,你不可以使用在if语句中,Python仍然没有#ifdef 宏。下面尝试把括号里面的参数打印出来:

>>> print('Hello World!')
Hello World!

很酷,这个在Python2和Python3中都可以运行,而且运行的效果是一样的,再来看看下面这段:
 

>>> print(10, 20) # Python 2
(10, 20)

此时,你并没有像前面那样幸运得到一样的结果,Python2中打印的是元组(tuple),而在Python3中传递多个参数到print()里面时打印的是两个值:
 

>>> print(10, 20) # Python 3
10 20

如果你思考得比较多的话,我们可以检查print是否是一个关键字,keyword模块包含一个关键字列表。print在3.x中不是关键字,可以简单验证一下:
 

>>> import keyword
>>> 'print' in keyword.kwlist
False

作为一名聪明的程序员,你可能在2.x中尝试的时候期待的结果是True,尽管这并没有错,但是为了达到Python3的效果,但你仍然会因为其他原因导致失败。
 

>>> import keyword
>>> if 'print' in keyword.kwlist:
...  from __future__ import print_function
...
File "", line 2
SyntaxError: from __future__ imports must occur at the beginning of the file

一种解决方案是使用一个函数,其功能类似于print,其中之一是sys.stdout.write(),另一个是distutils.log.warn()。不管出于什么原因,我们决定使用后者。“hello world”的例子看起来是这样的:
 

# Python 2.x
print 'Hello World!'
# Python 3.x
print('Hello World!')

下面的代码就可以在两个版本中通用:
 

# Python 2.x & 3.x compatible
from distutils.log import warn as printf
printf('Hello World!')

为什么我们不用sys.stdout.write()呢,因为我们需要添加一个NEWLINE字符在字符串的结尾来兼容这种行为(python2.x中write方法不会换行):
 

# Python 2.x & 3.x compatible
import sys
sys.stdout.write('Hello World!n')

Import your way to a solution

一般情况情况下,import时没什么烦恼,只要正确的导入就行,但在下面代码中,我们想导入urlopen()函数,在Python2中,他同时存在与urllib2和urllib2中(我们使用后者),在Python3中,他被集成到了urllib.request中,而你的方案是要既能在2.x和3.x中正常工作:

try:
 from urllib2 import urlopen
except ImportError:
 from urllib.request import urlopen

出于对内存的保护,也许你对iterator(Python3)版本的zip()更加有兴趣,在Python2中,iterator版本是itertools.izip()。这个函数在Python3中被重命名替换成了zip()。如果你使用迭代版本,导入语句也非常直白:
 

try:
 from itertools import izip as zip
except ImportError:
 pass

另一个列子是看来来并不怎么优雅的StringIO类,在Python2中,纯Python版本是StringIO模块,意味着访问的时候是通过StringIO.StringIO,同样还有一个更为快速的C语言版本,位于cStringIO.StringIO,不过这取决你的Python安装版本,你可以优先使用cStringIO然后是StringIO(如果cStringIO不能用的话)。在Python3中,Unicode是默认的string类型,但是如果你做任何和网络相关的操作,很有可能你不得不用ASCII/字节字符串来操作,所以代替StringIO,你要io.BytesIO,为了达到你想要的,这个导入看起来有点丑:
 

try:
 from io import BytesIO as StringIO
except ImportError:
 try:
  from cStringIO import StringIO
 except ImportError:
  from StringIO import StringIO

Putting it all together

如果你运气好的话,上面那些就是你要准备做的全部,剩下的代码都比开始设置的地方更简单。如果你按照上面的方式导入了distutils.log.warn()[printf()],url*urlopen(),*.StringIO和一个标准的导入:xml.etree.ElementTree(2.5及更新的),现在你就可以写一个非常简短短的解析器来展示从Google News服务中提供的头条故事(译注:当然首先得备一个梯子),只需八行代码:
 

g = urlopen('http://news.google.com/news?topic=h&output=rss')
f = StringIO(g.read())
g.close()
tree = xml.etree.ElementTree.parse(f)
f.close()
for elmt in tree.getiterator():
 if elmt.tag == 'title' and not
   elmt.text.startswith('Top Stories'):
  printf('- %s' % elmt.text)

这段脚本在2.x和3.x下面运行时,不需要做任何改动,运行效果完全一样,当然,如果你正在使用的是2.4或者更老的版本,你需要单独下载ElementTree。

但是有时候感觉这些改变把你优雅的Python代码弄得一团糟,毕竟可读性才是最重要的,如果你要优先保证代码的整洁而且在不修改任何地方的前提下运行在两个版本的Python环境中,那么你可以看一下six包。

six一个兼容库,它的主要任务是提供接口隐藏复杂的细节,你可以在这里找到它。无论你是使用像six这样的库还是用自己的方法来做,我们希望这个简短的介绍可以让你开始考虑写的代码能够在2.x和3.x下同时运行。

Python 相关文章推荐
python fabric实现远程部署
Jan 05 Python
Python 专题二 条件语句和循环语句的基础知识
Mar 19 Python
python2与python3中关于对NaN类型数据的判断和转换方法
Oct 30 Python
Python一句代码实现找出所有水仙花数的方法
Nov 13 Python
详解Python爬取并下载《电影天堂》3千多部电影
Apr 26 Python
详解python 爬取12306验证码
May 10 Python
python 实现二维字典的键值合并等函数
Dec 06 Python
pycharm设置python文件模板信息过程图解
Mar 10 Python
Pymysql实现往表中插入数据过程解析
Jun 02 Python
Python OpenCV去除字母后面的杂线操作
Jul 05 Python
Python调用REST API接口的几种方式汇总
Oct 19 Python
tensorboard 可视化之localhost:6006不显示的解决方案
May 22 Python
以Python的Pyspider为例剖析搜索引擎的网络爬虫实现方法
Mar 30 #Python
在树莓派2或树莓派B+上安装Python和OpenCV的教程
Mar 30 #Python
Python中利用函数装饰器实现备忘功能
Mar 30 #Python
利用Python绘制MySQL数据图实现数据可视化
Mar 30 #Python
Python面向对象编程中的类和对象学习教程
Mar 30 #Python
详细介绍Python函数中的默认参数
Mar 30 #Python
在Python中利用Into包整洁地进行数据迁移的教程
Mar 30 #Python
You might like
php网页病毒清除类
2014/12/08 PHP
深入讲解PHP Session及如何保持其不过期的方法
2015/08/18 PHP
WordPress中获取指定分类及其子分类下的文章数目
2015/12/31 PHP
一个背景云变换js特效 鼠标移动背景云变化
2012/12/28 Javascript
JQuery动画animate的stop方法使用详解
2014/05/09 Javascript
2014年最火的Node.JS后端框架推荐
2014/10/27 Javascript
jquery+CSS实现的水平布局多级网页菜单效果
2015/08/24 Javascript
JavaScript html5 canvas画布中删除一个块区域的方法
2016/01/26 Javascript
一篇文章让你彻底弄懂JS的事件冒泡和事件捕获
2017/08/14 Javascript
探索webpack模块及webpack3新特性
2017/09/18 Javascript
AngularJS监听ng-repeat渲染完成的方法
2018/03/20 Javascript
微信小程序蓝牙连接小票打印机实例代码详解
2019/06/03 Javascript
使用jquery实现轮播图效果
2021/01/02 jQuery
[58:09]Spirit vs NB Supermajor小组赛 A组败者组决赛 BO3 第三场 6.2
2018/06/03 DOTA
[53:03]Optic vs TNC 2018国际邀请赛小组赛BO2 第一场 8.17
2018/08/18 DOTA
基于scrapy实现的简单蜘蛛采集程序
2015/04/17 Python
python实现连接mongodb的方法
2015/05/08 Python
Python生成不重复随机值的方法
2015/05/11 Python
Python输出汉字字库及将文字转换为图片的方法
2016/06/04 Python
分享一个可以生成各种进制格式IP的小工具实例代码
2017/07/28 Python
Python中用psycopg2模块操作PostgreSQL方法
2017/11/28 Python
numpy找出array中的最大值,最小值实例
2018/04/03 Python
PyQt5每天必学之带有标签的复选框
2018/04/19 Python
在PyCharm下使用 ipython 交互式编程的方法
2019/01/17 Python
详解pandas的外部数据导入与常用方法
2019/05/01 Python
python 图片去噪的方法示例
2019/07/09 Python
Django中自定义admin Xadmin的实现代码
2019/08/09 Python
Python matplotlib修改默认字体的操作
2020/03/05 Python
html5+svg学习指南之SVG基础知识
2014/12/17 HTML / CSS
匡威意大利官方商店 :Converse意大利
2018/11/27 全球购物
《浅水洼里的小鱼》听课反思
2014/02/28 职场文书
yy婚礼主持词
2014/03/14 职场文书
合作意向协议书
2015/01/29 职场文书
SQL Server作业失败:无法确定所有者是否有服务器访问权限的解决方法
2021/06/30 SQL Server
Oracle以逗号分隔的字符串拆分为多行数据实例详解
2021/07/16 Oracle
win7配置本地ftp服务器的图文教程
2022/08/05 Servers