Python 2 和 Python 3 主要區別有哪些(一)?

Guido(Python之父,仁慈的獨裁者)在設計 Python3 的過程當中,受一篇文章 「Python warts」 的影響,決定不向後兼容,不然沒法修復大多數缺陷。---摘錄自《流暢的Python》html

你可能歷來沒有據說過學 Java 的糾結是學 JDK6 仍是 JDK7,也沒據說學 PHP 的糾結是學 PHP5 仍是 PHP7,但在 Python 社區,有這麼個怪問題:「學 Python 究竟是學2仍是學3?」這個問題就像月經同樣每隔斷時間就出如今你面前,也成了不少初學者的選擇困惑,這個問題的「始做俑者」固然是 Python 它爹,你們衆說紛紜,有說 Python2 是主流,大公司都在用,你應該學 2 。也有說 Python3 纔是將來主流,大多數第三方框架已基本支持 Python3。我的見解是 Python2 還會存在很長一段時間(只要那些用 Python2 的公司還沒倒閉,就一直會存在),你去找工做頗有可能就須要用到 2,而 Python3 也是你必需要掌握的,由於愈來愈多的人會遷移到 3 上去,本質上,它倆是同一門語言,僅僅只是極少部分(1%?並無嚴格統計)不兼容的地方,因此就沒所謂學哪個好,學了一個,另外一個花不多時間就能掌握。今天給你們介紹 Python2 和 Python3 的一些主要區別。python

print

在進行程序調試時用得最多的語句可能就是 print,在 Python 2 中,print 是一條語句,而 Python3 中做爲函數存在。有人可能就有疑問了,我在 Python2 中明明也看到當函數使用:bash

# py2
print("hello")  # 等價 print ("hello")

#py3
print("hello")複製代碼

然而,你看到的只是表象,那麼上面兩個表達式有什麼區別?從輸出結果來看是同樣的,但本質上,前者是把 ("hello")看成一個總體,然後者 print()是個函數,接收字符串做爲參數。網絡

# py2
>>> print("hello", "world")
('hello', 'world')

# py3
>>> print("hello", "world")
hello world複製代碼

這個例子更明顯了,在 py2 中,print語句後面接的是一個元組對象,而在 py3 中,print 函數能夠接收多個位置參數。若是但願在 Python2 中 把 print 當函數使用,那麼能夠導入 future 模塊 中的 print_function框架

# py2
>>> print("hello", "world")
('hello', 'world')
>>> 
>>> from __future__ import print_function
>>> print("hello", "world")
hello world複製代碼

編碼

Python2 的默認編碼是 asscii,這也是致使 Python2 中常常遇到編碼問題的緣由之一,至因而爲何會使用 asscii 做爲默認編碼,緣由在於 Python這門語言出來的時候還沒出現 Unicode。Python 3 默認採用了 UTF-8 做爲默認編碼,所以你再也不須要在文件頂部寫 # coding=utf-8 了。異步

# py2
>>> sys.getdefaultencoding()
'ascii'

# py3
>>> sys.getdefaultencoding()
'utf-8'複製代碼

網上很多文章說經過修改默認編碼格式來解決 Python2 的編碼問題,其實這是個大坑,不要這麼幹。函數

字符串

字符串是最大的變化之一,這個變化使得編碼問題降到了最低可能。在 Python2 中,字符串有兩個類型,一個是 unicode,一個是 str,前者表示文本字符串,後者表示字節序列,不過二者並無明顯的界限,開發者也感受很混亂,不明白編碼錯誤的緣由,不過在 Python3 中二者作了嚴格區分,分別用 str 表示字符串,byte 表示字節序列,任何須要寫入文本或者網絡傳輸的數據都只接收字節序列,這就從源頭上阻止了編碼錯誤的問題。大數據

py2 py3 表現 轉換 做用
str byte 字節 encode 存儲、傳輸
unicode str 字符 decode 展現

True和False

True 和 False 在 Python2 中是兩個全局變量(名字),在數值上分別對應 1 和 0,既然是變量,那麼他們就能夠指向其它對象,例如:ui

# py2
>>> True = False
>>> True
False
>>> True is False
True
>>> False = "x"
>>> False
'x'
>>> if False:
...     print("?")
... 
?複製代碼

顯然,上面的代碼違背了 Python 的設計哲學 Explicit is better than implicit.。而 Python3 修正了這個缺陷,True 和 False 變爲兩個關鍵字,永遠指向兩個固定的對象,不容許再被從新賦值。編碼

# py3
>>> True = 1
  File "<stdin>", line 1
SyntaxError: can't assign to keyword複製代碼

迭代器

在 Python2 中不少返回列表對象的內置函數和方法在 Python 3 都改爲了返回相似於迭代器的對象,由於迭代器的惰性加載特性使得操做大數據更有效率。Python2 中的 range 和 xrange 函數合併成了 range,若是同時兼容2和3,能夠這樣:

try:
    range = xrange
except:
    pass複製代碼

另外,字典對象的 dict.keys()、dict.values() 方法都再也不返回列表,而是以一個相似迭代器的 "view" 對象返回。高階函數 map、filter、zip 返回的也都不是列表對象了。Python2的迭代器必須實現 next 方法,而 Python3 改爲了 __next__

nonlocal

咱們都知道在Python2中能夠在函數裏面能夠用關鍵字 global 聲明某個變量爲全局變量,可是在嵌套函數中,想要給一個變量聲明爲非局部變量是無法實現的,在Pyhon3,新增了關鍵字 nonlcoal,使得非局部變量成爲可能。

def func():
    c = 1
    def foo():
        c = 12
    foo()
    print(c)
func()    #1複製代碼

能夠對比上面兩段代碼的輸出結果

def func():
    c = 1
    def foo():
        nonlocal c
        c = 12
    foo()
    print(c)
func()   # 12複製代碼

其實不少內建模塊也作了大量調整,Python3 中的模塊組織更加清晰,類更加先進,還引入了異步IO,此次先寫這麼多,下次再繼續。

原文:foofish.net/python2_pyt…

公衆號:Python之禪
公衆號:Python之禪
相關文章
相關標籤/搜索