前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >连仕彤博客[Python笔记] Python3增加的关键字nonlocal

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

作者头像
行 者
发布2018-06-13 14:42:42
4680
发布2018-06-13 14:42:42
举报
文章被收录于专栏:运维技术迷运维技术迷

前言

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
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 前言
  • 验证步骤
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档