python2 與 python3?? 版本區(qū)別
python 的 3??.0 版本,常被稱為 python 3000,或簡(jiǎn)稱 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è)方面:
1. 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() 語法,實(shí)例如下:
from __future__ import print_function print("fish", "panda", sep=', ')
如果 python2.x 版本想使用使用 python3.x 的 print 函數(shù),可以導(dǎo)入 __future__ 包,該包禁用 python2.x 的 print 語句,采用 python3.x 的 print 函數(shù):
>>> list =["a", "b", "c"] >>> print list ? ?# python2.x 的 print 語句 ['a', 'b', 'c'] >>> from __future__ import print_function ?# 導(dǎo)入 __future__ 包 >>> print list ? ? # python2.x 的 print 語句被禁用,使用報(bào)錯(cuò) ? file "<stdin>", line 1 ? ? print list ? ? ? ? ? ? ?^ syntaxerror: invalid syntax >>> print (list) ? # 使用 python3.x 的 print 函數(shù) ['a', 'b', 'c'] >>>
python3.x 與 python2.x 的許多兼容性設(shè)計(jì)的功能可以通過 __future__ 這個(gè)包來導(dǎo)入。
2. 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 編碼,所以使用中文就更加方便了:
>>> 中國(guó) = 'china' >>>print(中國(guó)) china
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 '我愛北京天安門'
3. 除法運(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
4. 異常
在 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ā)生的情況才能去用異常捕獲語句來處理。
5. 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
6. 八進(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
7. 不等運(yùn)算符
python 2.x中不等于有兩種寫法 != 和 <>
python 3.x中去掉了<>, 只有!=一種寫法,還好,我從來沒有使用<>的習(xí)慣
8. 去掉了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
9. 多個(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ù)。
10. 數(shù)據(jù)類型
1)py3.x去除了long類型,現(xiàn)在只有一種整型——int,但它的行為就像2.x版本的long
2)新增了bytes類型,對(duì)應(yīng)于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替代它吧 。