连仕彤博客[Python笔记] Python3增加的关键字nonlocal

前言

Python里只有2种作用域:全局作用域和局部作用域。全局作用域是指当前代码所在模块的作用域,局部作用域是指当前函数或方法所在的作用域。其实准确来说,Python 3.x引入了nonlocal关键字,可以用于标识外部作用域的变量。 局部作用域里的代码可以读外部作用域(包括全局作用域)里的变量,但不能更改它。一旦进行更改,就会将其当成是局部变量。而如果在更改前又进行了读取操作,则会抛出异常。

验证步骤

Python中(2.X和3.X版本)如果在一个函数的局部作用域中修改外部作用域的变量,就会报UnboundLocalError错误:

In [6]: help()
 
Welcome to Python 2.7!  This is the online help utility.
....
In [7]: def foo():
   ...:     c = 1
   ...:     def inner():
   ...:         c += 1
   ...:         return c
   ...:     return inner
   ...:
 
In [8]: print foo()()
---------------------------------------------------------------------------
UnboundLocalError                         Traceback (most recent call last)
<ipython-input-8-1ec86e0af5ba> in <module>()
----> 1 print foo()()
 
<ipython-input-7-8107f285e6fe> in inner()
      2     c = 1
      3     def inner():
----> 4         c += 1
      5         return c
      6     return inner
 
UnboundLocalError: local variable 'c' referenced before assignment

如果想要对外部作用域里面的变量进行修改,最简单的办法就是将其放入全局作用域,用global关键字引入该变量。

In [9]: def foo():
    ...:     global c
    ...:     c = 1
    ...:     def inner():
    ...:         global c
    ...:         c += 1
    ...:         return c
    ...:     return inner
 
In [10]: print foo()()
2

在Python 2.x版本中中,闭包只能读外部函数的变量,而不能改写它。

In [11]: def a():
    ...:     x =0
    ...:     def b():
    ...:         y = x +1
    ...:         print locals() # 返回当前位置的所有局部变量;locals()函数是只读的.
    ...:         print x,y
    ...:     return b
    ...:
 
In [12]: print a()()
{'y': 1, 'x': 0} # 可以看到返回值中,x变量的值并没有修改,也不能修改。
0 1
None

如果要对x进行赋值操作,在Python 2.x中解决这个问题,目前只能使用全局变量:global或者像上面的这个例子一样曲线救国。

为了解决这个问题,Python 3.x引入了nonlocal关键字(详见The nonlocal statement)。只要在闭包内用nonlocal声明变量,就可以让解释器在外层函数中查找变量名了.

In [13]: def a():
    ...:     x = 0
    ...:     def b():
    ...:         nonlocal x
    ...:         x += 1
    ...:         print(x)
    ...:     return b
In [14]: print(a()())
1
None

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏前端儿

大小写互换

  现在给出了一个只包含大小写字母的字符串,不含空格和换行,要求把其中的大写换成小写,小写换成大写,然后输出互换后的字符串。

1032
来自专栏Dato

关于 Integer 值比较的问题

今天刚好遇到这样的问题,别的不说,先上代码 public class TestInteger { public static void main(fi...

2708
来自专栏你不就像风一样

Java多线程编程核心技术(二)对象及变量的并发访问

本文主要介绍Java多线程中的同步,也就是如何在Java语言中写出线程安全的程序,如何在Java语言中解决非线程安全的相关问题。阅读本文应该着重掌握如下技术点:

923
来自专栏尚国

PHP反序列化漏洞

这里你可以看到, 我代码里的类定义为: class F, 这个序列化就是 F, 我定义变量名字是filename, 它这里也是 filename, 我们可以修改...

802
来自专栏C语言及其他语言

[蓝桥杯]字符逆序

题目描述 将一个字符串str的内容颠倒过来,并输出。str的长度不超过100个字符。 输入 输入包括一行。 第一行输入的字符串。 输出 输出转换好的逆序字符串。...

3435
来自专栏Hongten

java多线程系列_用Thread类创建线程(2)

在Java中创建线程有两种方法:使用Thread类和使用Runnable接口。在使用Runnable接口时需要建立一个Thread实例。因此,无论是通过Thre...

923
来自专栏xiaoxi666的专栏

strcpy和strcat易忽略点

1061
来自专栏老司机的技术博客

宝宝都能学会的python编程教程11:定义函数

定义函数 在Python中,定义一个函数要使用def语句,依次写出函数名、括号、括号中的参数和冒号:,然后,在缩进块中编写函数体,函数的返回值用return语句...

2855
来自专栏北京马哥教育

17个案例带你3分钟搞定Linux正则表达式

正则表达式是一种字符模式,用于在查找过程中匹配制定的字符。 元字符通常在Linux中分为两类: Shell元字符,由Linux Shell进行解析; 正则表达式...

3364
来自专栏用户2442861的专栏

深入 char * ,char ** ,char a[ ] ,char *a[] 内核

http://blog.csdn.net/daiyutage/article/details/8604720

992

扫码关注云+社区