作用域
在函数中,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
所在的作用域寻找。