最近闲着时就看看《Python核心编程》第二版,这本书很老了但是基础知识挺全的,自己一直用Python3实践,发现一个有趣的现象。
Python作用域与C/C++的区别
首先说到作为C++er刚接触Python时遇到一个疑惑的问题,就是类似下面这样的代码
if 1 < 2:
x = 1
else:
x = -1
print(x)
现代C++er的基本素质是时刻铭记某变量的生命周期,所以按照C++的变量离开作用域即销毁的思路,我会觉得这里的变量x已经被回收了。在if-else块之外打印x,相当于引用一个未定义的对象。
然而这段代码是正确的,因为python的作用域概念和C++不一样,只有def/class/lambda影响python变量的作用域,if-else块/try-except-finally块/for块/while块均不改变变量的作用域。
Python异常参数在2和3的作用域区别
然后实践10.3.6节的代码时发现python2和3在异常参数的生命周期上有区别
Python 2.7.12 (default, Nov 20 2017, 18:23:56)
[GCC 5.4.0 20160609] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> try:
... 1 / 0
... except Exception, e:
... pass
...
>>> type(e)
<type 'exceptions.ZeroDivisionError'>
Python 3.5.2 (default, Nov 23 2017, 16:37:01)
[GCC 5.4.0 20160609] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> try:
... 1 / 0
... except Exception as e:
... pass
...
>>> type(e)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: name 'e' is not defined
可以发现在python2中,except后接的异常参数e在try-except块之外可以访问,然而在python3中则不可以,提示e未被定义。
但是如果在except块中定义的变量是可以被外部访问的
>>> try:
... 1 / 0
... except Exception as e:
... e0 = e
... pass
...
>>> type(e0)
<class 'ZeroDivisionError'>
也就是说python3中try-except块仍然不改变变量的作用域,但是except后接的异常参数是默认对外部不可见的。