Python 2.x 與 3??.x 版本區(qū)別

2022-07-04 14:57 更新

Python 的 3??.0 版本,常被稱為 Python 3000,或簡稱 Py3k。相對(duì)于 Python 的早期版本,這是一個(gè)較大的升級(jí)。

為了不帶入過多的累贅,Python 3.0 在設(shè)計(jì)的時(shí)候沒有考慮向下相容。

許多針對(duì)早期 Python 版本設(shè)計(jì)的程式都無法在 Python 3.0 上正常執(zhí)行。

為了照顧現(xiàn)有程式,Python 2.6 作為一個(gè)過渡版本,基本使用了 Python 2.x 的語法和庫,同時(shí)考慮了向 Python 3.0 的遷移,允許使用部分 Python 3.0 的語法與函數(shù)。

新的Python程式建議使用 Python 3.0 版本的語法。

除非執(zhí)行環(huán)境無法安裝 Python 3.0 或者程式本身使用了不支援 Python 3.0 的第三方庫。目前不支援 Python 3.0 的第三方庫有 Twisted, py2exe, PIL等。

大多數(shù)第三方庫都正在努力地相容 Python 3.0 版本。即使無法立即使用 Python 3.0 ,也建議編寫相容Python 3.0版本的程式,然后使用 Python 2.6, Python 2.7 來執(zhí)行。

Python 3.0 的變化主要在以下幾個(gè)方面:


print 函數(shù)

?print?語句沒有了,取而代之的是?print()?函數(shù)。 Python 2.6 與 Python 2.7 部分地支持這種形式的?print?語法。在 Python 2.6 與 Python 2.7 里面,以下三種形式是等價(jià)的:

print "fish"
print ("fish") #注意print后面有個(gè)空格
print("fish") #print()不能帶有任何其它參數(shù)

然而,Python 2.6 實(shí)際已經(jīng)支持新的?print()?語法:

from __future__ import print_function
print("fish", "panda", sep=', ')

Unicode

Python 2 有 ?ASCII str()? 類型,?unicode()? 是單獨(dú)的,不是 ?byte? 類型。

現(xiàn)在, 在 Python 3,我們最終有了? Unicode (utf-8)? 字符串,以及一個(gè)字節(jié)類:?byte? 和 ?bytearrays?。

由于 Python3.X 源碼文件默認(rèn)使用?utf-8?編碼,這就使得以下代碼是合法的:

>>> 編程獅 = 'W3Cschool' 
>>>print(編程獅) 
W3Cschool

Python 2.x

>>> str = "我愛北京天安門"
>>> str
'\xe6\x88\x91\xe7\x88\xb1\xe5\x8c\x97\xe4\xba\xac\xe5\xa4\xa9\xe5\xae\x89\xe9\x97\xa8'
>>> str = u"我愛北京天安門"
>>> str
u'\u6211\u7231\u5317\u4eac\u5929\u5b89\u95e8'

Python 3.x

>>> str = "編程獅"
>>> str
'編程獅'

除法運(yùn)算

Python 中的除法較其它語言顯得非常高端,有套很復(fù)雜的規(guī)則。Python 中的除法有兩個(gè)運(yùn)算符,?/?和?//?

首先來說/除法:

在 python 2.x 中/除法就跟我們熟悉的大多數(shù)語言,比如 Java 啊 C 啊差不多,整數(shù)相除的結(jié)果是一個(gè)整數(shù),把小數(shù)部分完全忽略掉,浮點(diǎn)數(shù)除法會(huì)保留小數(shù)點(diǎn)的部分得到一個(gè)浮點(diǎn)數(shù)的結(jié)果。

在 python 3.x 中/除法不再這么做了,對(duì)于整數(shù)之間的相除,結(jié)果也會(huì)是浮點(diǎn)數(shù)。

Python 2.x:

>>> 1 / 2
0
>>> 1.0 / 2.0
0.5

Python 3.x:

>>> 1/2
0.5

而對(duì)于//除法,這種除法叫做?floor?除法,會(huì)對(duì)除法的結(jié)果自動(dòng)進(jìn)行一個(gè)?floor?操作,在 python 2.x 和 python 3.x 中是一致的。

python 2.x:

>>> -1 // 2
-1

python 3.x:

>>> -1 // 2
-1

注意的是并不是舍棄小數(shù)部分,而是執(zhí)行 ?floor? 操作,如果要截取整數(shù)部分,那么需要使用 ?math? 模塊的? trunc? 函數(shù)

python 3.x:

>>> import math
>>> math.trunc(1 / 2)
0
>>> math.trunc(-1 / 2)
0

異常

在 Python 3 中處理異常也輕微的改變了,在 Python 3 中我們現(xiàn)在使用 ?as? 作為關(guān)鍵詞。

捕獲異常的語法由 ?except exc?, ?var? 改為 ?except exc as var?。

使用語法?except (exc1, exc2) as var?可以同時(shí)捕獲多種類別的異常。 Python 2.6已經(jīng)支持這兩種語法。

  1. 在 2.x 時(shí)代,所有類型的對(duì)象都是可以被直接拋出的,在 3.x 時(shí)代,只有繼承自?BaseException?的對(duì)象才可以被拋出。
  2. 2.x ?raise?語句使用逗號(hào)將拋出對(duì)象類型和參數(shù)分開,3.x 取消了這種奇葩的寫法,直接調(diào)用構(gòu)造函數(shù)拋出對(duì)象即可。

在 2.x 時(shí)代,異常在代碼中除了表示程序錯(cuò)誤,還經(jīng)常做一些普通控制結(jié)構(gòu)應(yīng)該做的事情,在 3.x 中可以看出,設(shè)計(jì)者讓異常變的更加專一,只有在錯(cuò)誤發(fā)生的情況才能去用異常捕獲語句來處理。


xrange

在 Python 2 中? xrange() ?創(chuàng)建迭代對(duì)象的用法是非常流行的。比如:? for? 循環(huán)或者是列表/集合/字典推導(dǎo)式。

這個(gè)表現(xiàn)十分像生成器(比如。"?惰性求值?")。但是這個(gè) ?xrange-iterable? 是無窮的,意味著你可以無限遍歷。

由于它的惰性求值,如果你不得僅僅不遍歷它一次,?xrange() ?函數(shù) 比 ?range()? 更快(比如 ?for? 循環(huán))。盡管如此,對(duì)比迭代一次,不建議你重復(fù)迭代多次,因?yàn)樯善髅看味紡念^開始。

在 Python 3 中,?range() ?是像 ?xrange()? 那樣實(shí)現(xiàn)以至于一個(gè)專門的? xrange()? 函數(shù)都不再存在(在 Python 3 中? xrange()? 會(huì)拋出命名異常)。

import timeit

n = 10000
def test_range(n):
    return for i in range(n):
        pass

def test_xrange(n):
    for i in xrange(n):
        pass   

Python 2

print 'Python', python_version()

print '\ntiming range()' 
%timeit test_range(n)

print '\n\ntiming xrange()' 
%timeit test_xrange(n)

Python 2.7.6

timing range()
1000 loops, best of 3: 433 μs per loop


timing xrange()
1000 loops, best of 3: 350 μs per loop

Python 3

print('Python', python_version())

print('\ntiming range()')
%timeit test_range(n)

Python 3.4.1

timing range()
1000 loops, best of 3: 520 μs per loop
print(xrange(10))
---------------------------------------------------------------------------
NameError                                 Traceback (most recent call last)
<ipython-input-5-5d8f9b79ea70> in <module>()
----> 1 print(xrange(10))

NameError: name 'xrange' is not defined

八進(jìn)制字面量表示

八進(jìn)制數(shù)必須寫成?0o777?,原來的形式?0777?不能用了;二進(jìn)制必須寫成?0b111?。

新增了一個(gè)?bin()?函數(shù)用于將一個(gè)整數(shù)轉(zhuǎn)換成二進(jìn)制字串。 Python 2.6 已經(jīng)支持這兩種語法。

在 Python 3.x 中,表示八進(jìn)制字面量的方式只有一種,就是?0o1000?。

python 2.x

>>> 0o1000
512
>>> 01000
512

python 3.x

>>> 01000
  File "<stdin>", line 1
    01000
        ^
SyntaxError: invalid token
>>> 0o1000
512

不等運(yùn)算符

Python 2.x 中不等于有兩種寫法? !=? 和 ?<>?

Python 3.x 中去掉了?<>?, 只有?!=?一種寫法,還好,我從來沒有使用?<>?的習(xí)慣


去掉了repr表達(dá)式``

Python 2.x 中反引號(hào)?``?相當(dāng)于?repr?函數(shù)的作用

Python 3.x 中去掉了?``?這種寫法,只允許使用?repr?函數(shù),這樣做的目的是為了使代碼看上去更清晰么?不過我感覺用?repr?的機(jī)會(huì)很少,一般只在?debug?的時(shí)候才用,多數(shù)時(shí)候還是用?str?函數(shù)來用字符串描述對(duì)象。

def sendMail(from_: str, to: str, title: str, body: str) -> bool:
    pass

多個(gè)模塊被改名(根據(jù)PEP8)

舊的名字 新的名字
_winreg winreg
ConfigParser configparser
copy_reg copyreg
Queue queue
SocketServer socketserver
repr reprlib

?StringIO?模塊現(xiàn)在被合并到新的?io?模組內(nèi)。 ?new?, ?md5?, ?gopherlib?等模塊被刪除。 Python 2.6已經(jīng)支援新的?io?模組。

?httplib?, ?BaseHTTPServer?, ?CGIHTTPServer?, ?SimpleHTTPServer?, ?Cookie?, ?cookielib?被合并到?http?包內(nèi)。

取消了?exec?語句,只剩下?exec()?函數(shù)。 Python 2.6已經(jīng)支援?exec()?函數(shù)。


5.數(shù)據(jù)類型

1)Python 3.X 去除了?long?類型,現(xiàn)在只有一種整型——?int?,但它的行為就像 Python 2.X 版本的?long?

2)新增了?bytes?類型,對(duì)應(yīng)于 Python 2.X 版本的八位串,定義一個(gè)?bytes?字面量的方法如下:

>>> b = b'china' 
>>> type(b) 
<type 'bytes'> 

?str? 對(duì)象和? bytes ?對(duì)象可以使用 ?.encode() (str -> bytes)? 或 ?.decode() (bytes -> str)?方法相互轉(zhuǎn)化。

>>> s = b.decode() 
>>> s 
'china' 
>>> b1 = s.encode() 
>>> b1 
b'china' 

3)?dict?的?.keys()?、?.items? 和?.values()?方法返回迭代器,而之前的?iterkeys()?等函數(shù)都被廢棄。同時(shí)去掉的還有? dict.has_key()?,用? in?替代它吧 。


相關(guān)教程


以上內(nèi)容是否對(duì)您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號(hào)
微信公眾號(hào)

編程獅公眾號(hào)