作用域
在函数中,Python
从命名空间中寻找变量的顺序如下:
local function scope
enclosing scope
global scope
builtin scope
例子:
local 作用域
In [1]:
- def foo(a,b):
- c = 1
- d = a + b + c
这里所有的变量都在 local
作用域。
global 作用域
In [2]:
- c = 1
- def foo(a,b):
- d = a + b + c
这里的 c
就在 global
作用域。
global 关键词
使用 global
关键词可以在 local
作用域中修改 global
作用域的值。
In [3]:
- c = 1
- def foo():
- global c
- c = 2
- print c
- foo()
- print c
- 1
- 2
其作用是将 c
指向 global
中的 c
。
如果不加关键词,那么 local
作用域的 c
不会影响 global
作用域中的值:
In [4]:
- c = 1
- def foo():
- c = 2
- print c
- foo()
- print c
- 1
- 1
built-in 作用域
In [5]:
- def list_length(a):
- return len(a)
- a = [1,2,3]
- print list_length(a)
- 3
这里函数 len
就是在 built-in
作用域中:
In [6]:
- import __builtin__
- __builtin__.len
Out[6]:
- <function len>
class 中的作用域
Global | MyClass |
---|---|
var = 0 MyClass access_class |
var = 1 access_class |
In [7]:
- # global
- var = 0
- class MyClass(object):
- # class variable
- var = 1
- def access_class_c(self):
- print 'class var:', self.var
- def write_class_c(self):
- MyClass.var = 2
- print 'class var:', self.var
- def access_global_c(self):
- print 'global var:', var
- def write_instance_c(self):
- self.var = 3
- print 'instance var:', self.var
Global | MyClass | obj |
---|---|---|
var = 0 MyClass [access_class ] obj |
var = 1 access_class |
In [8]:
- obj = MyClass()
查询 self.var
时,由于 obj
不存在 var
,所以跳到 MyClass 中:
Global | MyClass | obj |
---|---|---|
var = 0 MyClass [access_class self ] obj |
var = 1 access_class |
In [9]:
- obj.access_class_c()
- class var: 1
查询 var
直接跳到 global
作用域:
Global | MyClass | obj |
---|---|---|
var = 0 MyClass [access_class self ] obj |
var = 1 access_class |
In [10]:
- obj.access_global_c()
- global var: 0
修改类中的 MyClass.var
:
Global | MyClass | obj |
---|---|---|
var = 0 MyClass [access_class self ] obj |
var = 2 access_class |
In [11]:
- obj.write_class_c()
- class var: 2
修改实例中的 var
时,会直接在 obj
域中创建一个:
Global | MyClass | obj |
---|---|---|
var = 0 MyClass [access_class self ] obj |
var = 2 access_class |
var = 3 |
In [12]:
- obj.write_instance_c()
- instance var: 3
In [13]:
- MyClass.var
Out[13]:
- 2
MyClass
中的 var
并没有改变。
词法作用域
对于嵌套函数:
In [14]:
- def outer():
- a = 1
- def inner():
- print "a =", a
- inner()
- outer()
- a = 1
如果里面的函数没有找到变量,那么会向外一层寻找变量,如果再找不到,则到 global
作用域。
返回的是函数的情况:
In [15]:
- def outer():
- a = 1
- def inner():
- return a
- return inner
- func = outer()
- print 'a (1):', func()
- a (1): 1
func() 函数中调用的 a
要从它定义的地方开始寻找,而不是在 func
所在的作用域寻找。