值得注意的是,修改一个对象并不是对一个名称赋值。 变量名解析:LEGB原则
对于一个def语句:
变量名引用分为三个作用域进行查找:首先是本地,之后是函数内(如果有的话),之后全局,最后是内置。L->E->G->B
Python除了def/class/lambda外,其他如:if/elif/else/ try/except for/while并不能改变作用域。定义在他们之内的变量,外部还是可以访问。
>>> if True:
... a = 'I am A'
print g #结果为1
值得注意的是,有时候想再函数内调用全局变量,疏忽了会报错,如下:
#file1
var = 1
def func():
print var
var = 200
func()
#file2
var = 1
def func():
var = var +1
return var
func()
#这两个函数都会报错UnboundLocalError: local variable 'var' referenced before assignment
上述两个函数都会报同样的错误:为赋值之前引用变量!为什么?在函数内部,解释器探测到变量var重新被赋值,所以var变成了局部变量,但是在被赋值之前就使用了var,便会出现这个错误。解决的方法是在函数内部添加globals var语句,但运行函数后全局的var也会被修改。
#file1
var = 1
def func():
global var
print var
var = 200
func() #结果为1
print var #全局变量var变为200
#file2
var = 1
def func():
global var
var = var +1
return var
print func() #结果为2 闭包Closure
闭包的定义:如果在一个内部函数里,对外部函数内(不是全局变量)进行引用,那么内部函数就被认为是闭包(closure)。
a = 1
def external():
global a
a = 200
print a
b =100
def internal():
print b
b = 200
return b
internal()
print b
print external()
#一样会报错,赋值前引用UnboundLocalError: local variable 'b' referenced before assignment
Python3中有关键字nonlocal可以解决这个问题,但在Python2中尽量不要尝试修改闭包中的变量。
关于闭包,还有一个坑:
from functools import wraps
def wrapper(log):
def external(F):
@wraps(F)
def internal(**kw):
if False:
log = 'modified'
print log
return internal
return external
@wrapper('first')
def abc():
pass
print abc()
也会出现 引用在赋值之前 的错误,原因是解释器探测到了 if False 中的重新赋值,所以不会去闭包的外部函数(Enclosing)中找变量,但 if Flase 不成立没有执行,所以便会出现此错误。除非你还需要else: log='var' 或者 if True 但这样添加逻辑语句就没了意义,所以尽量不要修改闭包中的变量。
好像用闭包无法实现计数器功能,因为在闭包内部count+=1就会出现在赋值前引用的错误(Python3用关键字nonlocal可以解决)
def counter(start):
count = [start]