一个例子带你搞懂python作用域中的global、nonlocal和local

在编程中,只要接触过函数的,我相信都理解什么是全局变量和局部变量,概念比较简单,这里就不做解释了。在python中,用global语句就能将变量定义为全局变量,但是最近又发现有个nonlocal,一时搞不太清楚,仔细琢磨之后才搞明白。

首先看看官方文档里面是怎么说的:

global 语句是作用于整个当前代码块的声明。 它意味着所列出的标识符将被解读为全局变量。

nonlocal语句会使得所列出的名称指向之前最近的包含作用域中绑定的除全局变量以外的变量。

注意划重点了,global就是全局变量,这没有问题。至于nonlocal,“之前”、”最近“和“除全局变量以外”,就是最核心的地方,接下来用一个例子说明一下。

下面的示例代码同样来自官方文档

def scope_test():
    def do_local():
        spam = "local spam"

    def do_nonlocal():
        nonlocal spam
        spam = "nonlocal spam"

    def do_global():
        global spam
        spam = "global spam"

    spam = "test spam"
    do_local()
    print("After local assignment:", spam)
    do_nonlocal()
    print("After nonlocal assignment:", spam)
    do_global()
    print("After global assignment:", spam)

scope_test()
print("In global scope:", spam)

我们可以看到,这是定义了一个scope_test(),然后它里面又分别定义了三个函数do_local()、do_nonlocal()和do_global()。

输出的结果是什么呢?感兴趣的同学先自己琢磨琢磨,我放一张图提示一下:
在这里插入图片描述
下面正式揭晓答案了,输出结果为:(高亮忽视掉)

After local assignment: test spam
After nonlocal assignment: nonlocal spam
After global assignment: nonlocal spam
In global scope: global spam

我们观察四个print函数,首先,do_local()里面的spam是局部变量(local),这是毫无疑问的,所以它是不能改变spam的内容的,调用它之后,因为之前spam="test spam"了,所以输出为仍然为"After local assignment: test spam"。

然后,do_nonlocal()里面的 spam,因为有nonlocal进行修饰,它的作用范围,就不是局部了,那是哪里呢?正如文档中提到的,“指向之前最近的包含作用域中绑定的除全局变量以外的变量”,在调用do_nonlocal()时,哪个之前的spam离它最近呢?很明显,是spam="test spam"这一个,所以这个spam就变为了spam = "nonlocal spam",后面输出自然就是“After nonlocal assignment: nonlocal spam”。

为了加深理解,我们看看如果把spam = "test spam"搬到函数外面会怎么样。

spam = "test spam"

def scope_test():
    def do_local():
        spam = "local spam"

    def do_nonlocal():
        nonlocal spam
        spam = "nonlocal spam"

    def do_global():
        global spam
        spam = "global spam"

    do_local()
    print("After local assignment:", spam)
    do_nonlocal()
    print("After nonlocal assignment:", spam)
    do_global()
    print("After global assignment:", spam)

scope_test()
print("In global scope:", spam)

此时会报错:

 File "<ipython-input-1-3e4be319dd22>", line 7
    nonlocal spam
    ^
SyntaxError: no binding for nonlocal 'spam' found

这就充分说明了,nonlocal是不能指向全局变量的。

接着,do_global()里面的spam,很明显就是全局变量,对它进行赋值spam = "global spam",是对全局变量赋值。那影响print("After global assignment:", spam)吗?不影响,因为这句print中的spam是最开始spam = "test spam"中的那个,它被do_nonlocal()赋值为“nonlocal spam”,注意它可不是全局变量,因为在函数scope_test()中。所以说,此spam非彼spam,一个是全局变量,另一个虽然相对于小函数来说,是非局部变量,但毕竟也是大函数的局部变量呀,所以第三个print自然为“After global assignment: nonlocal spam”。

那到了最后,就简单了,因为调用完了函数scope_test(),里面只有do_global()改变了全局变量spam = "global spam",所以自然最后输出“In global scope: global spam"。

总结来说,全局变量global其实是个绝对概念,一旦使用,就会在整个代码文件适用;而local和nonlocal,无非是相对概念,就好像我们每个人自己可能都是学生(local),活到老学到老,只不过相对其他人来说,有的人是老师(非学生nonlocal)而已。

希望这篇文章能帮助大家理解。

  • 7
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Pythonglobalnonlocal可以让你在函数内部使用和更改全局变量和外部作用域的变量。Globalglobal用于在函数内部声明全局变量,即使在函数内部也可以访问该变量。例如:x = "global" def foo(): global x x = "local" print("x inside :", x) foo() print("x outside:", x)结果:x inside : local x outside: localnonlocalnonlocal用于在函数内部修改外部作用域的变量,即使在函数内部也可以访问和修改该变量。例如:x = "global" def outer(): x = "outer" def inner(): nonlocal x x = "inner" print("x inside :", x) inner() print("x outside:", x) outer()结果:x inside : inner x outside: inner ### 回答2: globalnonlocal都是python用于在函数内部访问外部变量的关键字。 首先,我们来看一下global关键字的用法。当我们在函数内部想要修改全局变量时,需要使用global关键字。下面是一个例子: ```python x = 10 def modify_global(): global x x = 20 modify_global() print(x) # 输出为20 ``` 在这个例子,通过在函数定义内使用`global x`,我们告诉python在函数内部修改的`x`是全局变量`x`,而不是在函数内部定义的局部变量。 接下来,我们来看一下nonlocal关键字的用法。当我们在嵌套函数想要修改外部函数的局部变量时,需要使用nonlocal关键字。下面是一个例子: ```python def outer(): x = 10 def inner(): nonlocal x x = 20 inner() print(x) # 输出为20 outer() ``` 在这个例子,通过在内部函数定义内使用`nonlocal x`,我们告诉python在内部函数修改的`x`是外部函数`outer()`的局部变量,而不是在内部函数定义的局部变量。 总结一下,global关键字用于在函数修改全局变量,nonlocal关键字用于在嵌套函数修改外部函数的局部变量。 ### 回答3: global关键字和nonlocal关键字都是用于访问外部作用域的变量,但它们之间有一些关键的区别。 首先,我们先了解一下global的用法。当我们在函数内部定义一个变量时,默认情况下它是局部变量,只在函数内部有效。如果需要在函数内部改变一个全局变量的值,我们就需要用到global关键字。下面是一个示例: ```python count = 0 def increment(): global count count += 1 print(count) # 输出0 increment() print(count) # 输出1 ``` 在上面的例子,我们声明了一个全局变量count,并定义了一个函数increment来增加count的值。在函数内部,我们使用global关键字告诉Python,count是一个全局变量,而不是一个局部变量。这样,在我们调用increment函数后,count的值会被成功增加。 接下来,我们来看一下nonlocal的用法。nonlocal关键字用于在嵌套函数访问上一级非全局作用域的变量。在Python,我们可以在函数内部定义另一个函数,这就是嵌套函数。下面是一个嵌套函数的例子: ```python def outer(): x = "hello" def inner(): nonlocal x x = "world" print("inner:", x) inner() print("outer:", x) outer() ``` 在上面的例子,我们定义了一个外部函数outer,它内部定义了一个局部变量x。然后,我们定义了一个嵌套函数inner,在inner函数内部,我们使用nonlocal关键字告诉Python,x是在外部函数定义的变量。这样,在我们调用inner函数后,x的值会被成功改变。运行上面的代码,会输出"inner: world"和"outer: world"。 综上所述,global用于访问全局作用域的变量,而nonlocal用于访问上一级非全局作用域的变量。使用这两个关键字能够更灵活地操作不同作用域的变量。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值