Python同時兼容python2和python3的8個技巧分享


  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讀物。

 


免責聲明!

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



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