Python 2 和 Python 3 主要区别有哪些(一)

Guido(Python之父,仁慈的独裁者)在设计 Python3 的过程中,受一篇文章 “Python warts” 的影响,决定不向后兼容,否则无法修复大多数缺陷。

—-摘录自《流畅的Python》

http://web.archive.org/web/20031002184114/www.amk.ca/python/writing/warts.html

你可能没听说过学 Java 纠结是学 JDK6 还是 JDK7,也没听说学 PHP 纠结是学 PHP5 还是 PHP7,但在 Python 社区,有这么个怪现象:“学 Python 到底是学 2 还是学 3?”,就像月经一样每隔断时间就出现在你面前,也成了很多初学者的选择困惑,问题的“始作俑者”当然是 Python 它爹,大家众说纷纭,有说 Python2 是主流,大公司都在用,你应该学 2 。也有说 Python3 才是未来主流,大多数第三方框架已基本支持 Python3。

个人看法是 Python2 还会存在很长一段时间(只要那些用 Python2 的公司还没倒闭,就一直会存在),你去找工作很有可能就需要用到 2,而 Python3 也是你必须要掌握的,因为越来越多项目会优先选择 3 ,本质上,它俩是同一门语言,仅仅只是极少部分(1%?并没有严格统计)不兼容的地方,所以就没所谓学谁好,学了一个,另一个花很少时间就能掌握。

今天就给大家介绍 Python2 和 Python3 的一些主要区别。

print

程序调试时用得最多的语句可能就是 print,在 Python2 中,print 是一条语句,而在 Python3 中是作为函数存在的。有人可能就有疑问了,我在 Python2 中明明也看到当函数使用:

 
 
 
 
  1. # py2 
  2. print("hello")  # 等价 print  ("hello") 
  3.  
  4. #py3 
  5. print("hello") 

然而,你看到的只是表象,上面两个表达式有什么区别?从输出结果来看是一样的,但实质上,前者是把 ("hello")当作一个整体,而后者 print()是个函数,接收字符串作为参数。

 
 
 
 
  1. # py2 
  2. >>> print("hello", "world") 
  3. ('hello', 'world') 
  4.  
  5. # py3 
  6. >>> print("hello", "world") 
  7. hello world 

这个例子就更明显了,在 py2 中,print语句后面接的是一个元组对象,而在 py3 中,print 函数可以接收多个位置参数。如果希望在 py2 中 把 print 当函数使用,那么可以导入 future 模块 中的 print_function

 
 
 
 
  1. # py2 
  2. >>> print("hello", "world") 
  3. ('hello', 'world') 
  4. >>>  
  5. >>> from __future__ import print_function 
  6. >>> print("hello", "world") 
  7. hello world 

编码

Python2 的默认编码是 asscii,这也是导致 Python2 中经常遇到编码问题的原因之一,至于是为什么会使用 asscii 作为默认编码,原因在于 Python 2 出来的时候还没出现 Unicode。Python 3 默认采用了 UTF-8 作为默认编码,因此你不再需要在文件顶部写 # coding=utf-8 了。

 
 
 
 
  1. # py2 
  2. >>> sys.getdefaultencoding() 
  3. 'ascii' 
  4.  
  5. # py3 
  6. >>> sys.getdefaultencoding() 
  7. 'utf-8' 

网上不少文章说通过修改默认编码格式来解决 Python2 的编码问题,其实这是个大坑,不要这么干。

字符串

字符串是***的变化之一,这个变化使得编码问题降到了***可能。在 Python2 中,字符串有两个类型,一个是 unicode,一个是 str,前者表示文本字符串,后者表示字节序列,不过两者并没有明显的界限,开发者也感觉很混乱,不明白编码错误的原因,不过在 Python3 中两者做了严格区分,分别用 str 表示字符串,byte 表示字节序列,任何需要写入文本或者网络传输的数据都只接收字节序列,这就从源头上阻止了编码错误的问题。

True和False

True 和 False 在 Python2 中是两个全局变量(名字),在数值上分别对应 1 和 0,既然是变量,那么他们就可以指向其它对象,例如:

 
 
 
 
  1. # py2 
  2. >>> True = False 
  3. >>> True 
  4. False 
  5. >>> True is False 
  6. True 
  7. >>> False = "x" 
  8. >>> False 
  9. 'x' 
  10. >>> if False: 
  11. ...     print("?") 
  12. ...  

显然,上面的代码违背了 Python 的设计哲学 Explicit is better than implicit.。而 Python3 修正了这个缺陷,True 和 False 变为两个关键字,永远指向两个固定的对象,不允许再被重新赋值。

 
 
 
 
  1. # py3 
  2. >>> True = 1 
  3.   File "", line 1 
  4. SyntaxError: can't assign to keyword 

迭代器

在 Python2 中很多返回列表对象的内置函数和方法在 Python 3 都改成了返回类似于迭代器的对象,因为迭代器的惰性加载特性使得操作大数据更有效率。Python2 中的 range 和 xrange 函数合并成了 range,如果同时兼容2和3,可以这样:

 
 
 
 
  1. try: 
  2.     range = xrange 
  3. except: 
  4.     pass 

另外,字典对象的 dict.keys()、dict.values() 方法都不再返回列表,而是以一个类似迭代器的 “view” 对象返回。高阶函数 map、filter、zip 返回的也都不是列表对象了。有,py2的迭代器必须实现 next 方法,而 py3 改成了 __next__

nolocal

我们都知道在 py2 中可以在函数里面可以用关键字 global声明某个变量为全局变量,但是在嵌套函数中,想要给一个变量声明为非局部变量是没法实现的,py3 新增了关键字 nolcoal,使得非局部变量成为可能。

 
 
 
 
  1. def func(): 
  2.     c = 1 
  3.     def foo(): 
  4.         c = 12 
  5.     foo() 
  6.     print(c) 
  7. func()    #1 

可以对比上面两段代码的输出结果

 
 
 
 
  1. def func(): 
  2.     c = 1 
  3.     def foo(): 
  4.         nonlocal c 
  5.         c = 12 
  6.     foo() 
  7.     print(count) 
  8. func()   # 12 

其实很多内建模块也做了大量调整,Python3 中的模块组织更加清晰,类更加先进,引入了异步IO,这次先写这么多,下次再继续。

【本文是专栏作者“刘志军”的原创文章,作者微信公众号:Python之禅(VTtalk)】

戳这里,看该作者更多好文

名称栏目:Python 2 和 Python 3 主要区别有哪些(一)
分享路径:http://www.csdahua.cn/qtweb/news27/371127.html

网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网