您的位置:首页 > 编程语言 > Python开发

让python同时兼容python2和python3的8个技巧分享

2017-07-10 10:24 701 查看
  python3和python2虽然不兼容,但他们之间差别并没很多人想像的那么大。你只需要对自己的代码稍微做些修改就可以很好的同时支持python2和python3的。下面我将简要的介绍一下如何让自己的python代码如何同时支持python2和python3。 
1、放弃python 2.6之前的python版本

  python 2.6之前的python版本缺少一些新特性,会给你的迁移工作带来不少麻烦。如果不是迫不得已还是放弃对之前版本的支持吧。
2、使用 2to3 工具对代码检查

  2to3是python自带的一个代码转换工具,可以将python2的代码自动转换为python3的代码。当然,不幸的是转换出的代码并没有对python2的兼容做任何的处理。所以我们并不真正使用2to3转换出的代码。执行2to3 t.py 查看输出信息,并修正相关问题。

脚本位置:D:\Python27\Tools\Scripts\2to3.py
案例:
待转换脚本:HouseStark.py
转化方法:D:\Python27\Tools\Scripts\2to3.py -w HouseStark.py
RefactoringTool: Skipping optional fixer: buffer
RefactoringTool: Skipping optional fixer: idioms
RefactoringTool: Skipping optional fixer: set_literal
RefactoringTool: Skipping optional fixer: ws_comma
RefactoringTool: Refactored HouseStark.py
--- HouseStark.py       (original)
+++ HouseStark.py       (refactored)
@@ -4,8 +4,8 @@
import os
import sys
import json
-import urllib
-import urllib2   # Python2
+import urllib.request, urllib.parse, urllib.error
+import urllib.request, urllib.error, urllib.parse   # Python2
import datetime
# import urllib.parse       Python3
# import urllib.request     Python3
@@ -70,29 +70,29 @@
url = "http://%s%s" % (settings.Params['server'], settings.Parms['urls'][action_type])
url = self.__attach_token(url)
-            print('Connecting [%s], it may take a minute' % url)
+            print(('Connecting [%s], it may take a minute' % url))
if method == "get":
args = ""
-                for k, v in data.items():
+                for k, v in list(data.items()):
args += "&%s=%s" % (k, v)
args = args[1:]
url_with_args = "%s?%s" % (url, args)
try:
-                    req = urllib2.Request(url_with_args)
-                    req_data = urllib2.urlopen(req, timeout=settings.Params['request_timeout'])
+                    req = urllib.request.Request(url_with_args)
+                    req_data = urllib.request.urlopen(req, timeout=settings.Params['request_timeout'])
callback = req_data.read()
-                    print("-->server response:", callback)
+                    print(("-->server response:", callback))
return callback
except urllib.URLError as e:
sys.exit("\033[31;1m%s\033[0m" % e)
elif method == "post":
try:
-                    data_encode = urllib.urlencode(data)
-                    req = urllib2.Request(url=url, data=data_encode.encode('utf-8'))
-                    res_data = urllib2.urlopen(req, timeout=settings.Params['request_timeout'])
+                    data_encode = urllib.parse.urlencode(data)
+                    req = urllib.request.Request(url=url, data=data_encode.encode('utf-8'))
+                    res_data = urllib.request.urlopen(req, timeout=settings.Params['request_timeout'])
callback = str(res_data.read(), encoding="utf-8")  # 调整编码
callback = json.loads(callback)
-                    print("\033[31;1m[%s]:[%s]\033[0m response:\n%s" % (method, url, callback))
+                    print(("\033[31;1m[%s]:[%s]\033[0m response:\n%s" % (method, url, callback)))
return callback
except Exception as e:
sys.exit("\033[31;1m%s\033[0m" % e)
RefactoringTool: Files that were modified:
RefactoringTool: HouseStark.py

查看原来的脚本已经转换为Python3代码,修改前脚本会自动备份脚本
查看更多帮助,使用--help参数,例如:
1、如果上述不加-w参数,则默认只是把转换过程所对应的diff内容打印输出到当前窗口而已。
2、加了-w,就是把改动内容,写回到原先的文件了。
3、不想要生成bak文件,再加上-n即可。
4、不想看到那一堆输出的内容,加上–no-diffs,即可。


 
3、使用python -3执行python程序

  2to3 可以检查出很多python2&3的兼容性问题,但也有很多问题是2to3发现不了的。在加上 -3 参数后,程序在运行时会在控制台上将python2和python3不一致,同时2to3无法处理的问题提示出来。比如python3和python2中对除法的处理规则做过改变。使用-3参数执行4/2将提示 DeprecationWarning: classic int division 。
4、from __future__ import

  “from __future__ import”后即可使使用python的未来特性了。python的完整future特性可见 __future__ 。python3中所有字符都变成了unicode。在python2中unicode字符在定义时需要在字符前面加 u,但在3中则不需要家u,而且在加u后程序会无法编译通过。为了解决该问题可以 “from future import unicode_literals” ,这样python2中字符的行为将和python3中保持一致,python2中定义普通字符将自动识别为unicode。
5、import问题

  python3中“少”了很多python2的包,在大多情况下这些包之是改了个名字而已。我们可以在import的时候对这些问题进行处理。

try:#python2
from UserDict import UserDict
#建议按照python3的名字进行import
from UserDict import DictMixin as MutableMapping

except ImportError:#python3
from collections import UserDict
from collections import MutableMapping


 
6、使用python3的方式写程序

  python2中print是关键字,到了python3中print变成了函数。事实上在python2.6中已经带了print函数,所以对print你直接按照2to3中给出的提示改为新写法即可。在python3中对异常的处理做了些变化,这个和print类似,直接按照2to3中的提示修改即可。
7、检查当前运行的python版本

  有时候你或许必须为python2和python3写不同的代码,你可以用下面的代码检查当前系统的python版本。

import sys
if sys.version > '3':
PY3 = True
else:
PY3 = False


8、six

  six 提供了一些简单的工具用来封装 Python 2 和 Python 3 之间的差异性。我并不太推荐使用six。如果不需要支持python2.6之前的python版本,即使不用six也是比较容易处理兼容性问题的。使用six会让你的代码更像python2而不是python3。

python3的普及需要每位pythoner的推动,或许你还无法立即升级到python3,但请现在就开始写兼容python3的代码,并在条件成熟时升级到python3。
注:python2同python3的差异

如果你更全面的了解从python2迁移到python3的相关问题,推荐阅读 Porting to Python 3 这是一本免费的python读物。
 

出处:http://www.cnblogs.com/madsnotes/
声明:本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: