python3的作用域和C/C++有所不同
1、 按照L -> E -> G -> B的顺序查找变量
L (Local) 局部作用域
E (Enclosing) 函数外上一层的函数中
G (Global) 全局作用域
B (Built-in) 内建作用域
2、只有在模块module,类class,函数def、lambda中才可以改变作用域,其他情况不可。
-
例1
def func(): x= 100 print(x) func() print(x)
报错,因为第二个print(x)找不到变量x
-
例2
for i in range(4): x = i print(x)
正确,因为for语句并不能隔离作用域
-
例3
x= 1 def func(): print x x= 2 print func()
报错
UnboundLocalError: local variable 'x' referenced before assignment
,因为虽然python是逐行解释的。但每个函数属于一个区块,这个区块范围是一次性解释的,并不会读一行忘记一行,而是一直读,读完整个区块再解释。python读完整个fun区块后,检测到定义了x,所以x是局部变量。但是print x的时候会去寻找局部变量x,此时x还没被赋值,所以报错。详见python3代码块 -
例4
x= 1 def func(): x += 1 return x print func()
报错
UnboundLocalError: local variable 'x' referenced before assignment
,因为x+=1实际上是x=x+1,函数内部只要是赋值操作就表示声明为局部变量,所以x是局部变量。所以出现了为赋值前就使用x,所以报错 -
例5
x= "abc" def func1(): print(x) def func2(): x= "def" func1() func2()
输出abc,函数的定义位置决定了它所看见的变量,所以当调用func1()时,func1出于全局域中,会输出abc。
-
例6
x=3 def func1(): x=4 def func2(): print(x) x=5 func2() func1()
输出5,因为func2定义位置决定了看见的是func1的x,而在调用func2之前,func1的x已经赋值为5,故输出5
-
例7
x=3 def func1(): x=4 def func2(x=x): x += 3 print(x) x=5 func2() func1()
输出7,这里使用了默认参数,而默认参数是在生成函数而非调用函数时查找的,这里x=x,在执行到def的时候就引用为4了,但是其他参数,是在函数调用时查找的。
3、nonlocal默认将内层函数中的变量修饰为上一层函数的作用域范围,如果上一层函数中不存在该变量,则修饰为上上层、上上上层直到顶层函数,但不能修饰为全局作用域范围。
4、global可以声明一个或多个变量为全局变量,多个变量使用逗号隔开,也可以声明事先不存在的变量为全局变量,但nonlocal修饰的变量必须已经存在于上层或上上层(或更多层)函数,不能只存在于全局。
参考链接:https://www.cnblogs.com/f-ck-need-u/p/9925021.html#python%E4%BD%9C%E7%94%A8%E5%9F%9F%E8%A7%84%E5%88%99%E7%AE%80%E4%BB%8B