python 闭包变量不允许write,要使用nonlocal
以下是一段简单的闭包代码示例:
def foo(): m=3 n=5 def bar(): a=4 return m+n+a return bar >>>bar = foo() >>>bar() 12
是可以的!但是:
def foo():
m=3
n=5
def bar():
a=4
m += 1 #不可以!!!
return m+n+a
return bar
>>>bar = foo()
>>>bar()
12
python中global 和 nonlocal 的作用域
python引用变量的顺序: 当前作用域局部变量->外层作用域变量->当前模块中的全局变量->python内置变量 。
一 global
global关键字用来在函数或其他局部作用域中使用全局变量。但是如果不修改全局变量也可以不使用global关键字。
1 gcount = 0
2
3 def global_test():
4 gcount+=1
5 print (gcount)
6 global_test()
D:\Python34\python.exe E:/PycharmProjects/Day3/globaltest.py
Traceback (most recent call last):
File "E:/PycharmProjects/Day3/globaltest.py", line 6, in <module>
global_test()
File "E:/PycharmProjects/Day3/globaltest.py", line 4, in global_test
gcount+=1
UnboundLocalError: local variable 'gcount' referenced before assignment
Process finished with exit code 1
第一行定义了一个全局变量,(可以省略global关键字)。
在global_test 函数中程序会因为“如果内部函数有引用外部函数的同名变量或者全局变量,并且对这个变量有修改.那么python会认为它是一个局部变量,又因为函数中没有gcount的定义和赋值,所以报错。
二、声明全局变量,如果在局部要对全局变量修改,需要在局部也要先声明该全局变量:
1
2
3
4
5
6
7
|
gcount = 0 def global_test(): global gcount gcount + = 1 print (gcount) global_test() |
如果在函数中声明 gcount 是全局变量,即可对其进行修改。 正确输出 1 。
三、 在局部如果不声明全局变量,并且不修改全局变量。则可以正常使用全局变量:
1
2
3
4
5
|
gcount = 0 def global_test(): print (gcount) global_test() |
如果在局部不修改全局变量,程序正确输出 0 。
四、nonlocal关键字用来在函数或其他作用域中使用外层(非全局)变量。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
def make_counter(): count = 0 def counter(): nonlocal count count + = 1 return count return counter def make_counter_test(): mc = make_counter() print (mc()) print (mc()) print (mc()) make_counter_test() |
输出:
1
2
3
五、
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
def scope_test(): def do_local(): spam = "local spam" #此函数定义了另外的一个spam字符串变量,并且生命周期只在此函数内。此处的spam和外层的spam是两个变量,如果写出spam = spam + “local spam” 会报错 def do_nonlocal(): nonlocal spam #使用外层的spam变量 spam = "nonlocal spam" def do_global(): global spam spam = "global spam" spam = "test spam" do_local() print ( "After local assignmane:" , spam) do_nonlocal() print ( "After nonlocal assignment:" ,spam) do_global() print ( "After global assignment:" ,spam) scope_test() print ( "In global scope:" ,spam) |
输出是:
After local assignmane: test spam
After nonlocal assignment: nonlocal spam
After global assignment: nonlocal spam
In global scope: global spam
在函数 add_b 内 global 定义的变量 b,只能在 函数 do_global 内引用, 如果要在 do_global 内修改,必须在 do_global 函数里面声明 global b ,表明是修改外面的 全局变量 b :
def add_b():
global b
b = 42
def do_global():
global b
b = b + 10
print(b)
do_global()
print(b)
add_b()
global 定义的变量,表明其作用域在局部以外,即局部函数执行完之后,不销毁 函数内部以global定义的变量:
def add_a():
global a
a = 3
add_a()
print(a)
输出 3 。
def add_b():
global b
b = 42
def do_global():
global b
b = b + 10
print(b)
do_global()
print(b)
add_b()
print(b)
以上代码输出:
52
52
52
def add_b():
global b
b = 42
def do_global():
#global b
b = b + 10
print(b)
do_global()
print(b)
add_b()
以上代码报错:
Traceback (most recent call last):
File "E:/PycharmProjects/OOP/exe1.py", line 42, in <module>
add_b()
File "E:/PycharmProjects/OOP/exe1.py", line 40, in add_b
do_global()
File "E:/PycharmProjects/OOP/exe1.py", line 38, in do_global
b = b + 10
UnboundLocalError: local variable 'b' referenced before assignment
原因: global 定义的 b ,只能引用,不能修改。
def add_b():
global b
b = 42
def do_global():
global a
a = b + 10
print(b)
do_global()
print(a)
add_b()
print("a = %s , b = %s " %(a, b))
输出:
42
52
a = 52 , b = 42
def add_b():
#global b
b = 42
def do_global():
global b
b = 10
print(b)
do_global()
print(b)
add_b()
print(" b = %s " % b)
以上代码输出:
10
42
b = 10
def add_b():
#global b
b = 42
def do_global():
nonlocal b
b = 10
print(b)
do_global()
print(b)
add_b()
以上代码输出:
10
10
1
2
3
4
5
6
7
8
9
10
11
|
def add_b(): #global b b = 42 def do_global(): nonlocal b b = 10 print (b) do_global() print (b) add_b() print ( " b = %s " % b) |
以上代码报错:
print(" b = %s " % b)
NameError: name 'b' is not defined
说明: nonlocal 适用于在局部函数 中 的局部函数, 把最内层的局部 变量设置成外层局部可用,但是还不是全局的。
def add_b():
#global b
#b = 42
def do_global():
nonlocal b
b = 10
print(b)
do_global()
#print(b)
add_b()
以上代码报错:
File "E:/PycharmProjects/OOP/exe1.py", line 37
nonlocal b
SyntaxError: no binding for nonlocal 'b' found
nonlocal 要绑定一个局部变量。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 记一次.NET内存居高不下排查解决与启示
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
· DeepSeek 开源周回顾「GitHub 热点速览」
2017-03-24 lucene倒排索引瘦身的一些实验——merge的本质是减少cfx文件 变为pos和doc;存储term vector多了tvx和tvd文件有337M