Python3学习笔记 | 十九、Python的函数-作用域

  • 2019 年 10 月 6 日
  • 筆記

部分设备阅读本文会存在代码错乱的情况,可点击阅读原文链接到博客中进行查看

一、Python作用域基础

作用域针对的是变量。在使用同一个变量名时,会出现作用域问题。如右边的x,在函数里与函数外是不同的变量。 • 一个def内定义的变量名能够被def内部使用。不能在函数外部引用这个变量名。 • def内的变量名与def外的变量名并不冲突。在def内赋值的与def外赋值的相同变量名是不同的变量。 变量名有如下三种不同作用域: 1、全局:在def外定义的变量名是全局变量 2、本地:在def内部定义的叫做本地变量 3、其它:嵌套def时,各自的变量也是独立的。

二、作用域法则

内嵌的模块是全局作用域,全局作用域的作用范围仅限于单个文件,每次的函数的调用都创建了一个新的本地作用域,赋值的变量名除非声明全局变量或非本地变量(Python3.x有非本地变量说法),否则均为本地变量。所有其它的变量名都可以归纳为本地、全局或者内置。

变量名使用时,查找顺序:LEGB L:本地变量名 - 在本地是否使用此变量名赋值过。 E:上一层结构中def或lambda的本地变量名 - 上一层是否使用此变量名赋值过。 G:全局变量名 - 在整个文件里是否对变量名赋值过。 B:内置变量名 - Python内部是否存在此变量名。 要是都找不到相应变量名,就会报错。

内置变量:exit、open等。  >>> l = 'global'; e = 'global'; g = 'global'  #全局变量  >>>def myenclosefunc():  ...     e = 'enclose';l = 'enclose'  #enclose func变量  ...     def mylocalfunc():  ...             l = 'local'  #本地变量  ...             print('in local: l={},e={},g={}'.format(l,e,g))  ...     mylocalfunc()  ...     print('in enclose: l={},e={},g={}'.format(l,e,g))  ...  >>> myenclosefunc()  in local: l=local,e=enclose,g=global  in enclose: l=enclose,e=enclose,g=global  >>> print('in global: l={},e={},g={}'.format(l,e,g))  in global: l=global,e=global,g=global

三、global语句

在函数内,想改变全局变量,可以使用global语句来定义此变量为全局变量,如下所示。

>>> g = 'global'  >>> l = 'global'  >>> def myfunc():  ...     global g  ...     g = 'local'  ...     l = 'local'  ...  >>> g,l  ('global', 'global')  >>> myfunc()  >>> g,l  ('local', 'global')

在myfunc函数里,都重新赋值了g与l,但在函数执行后,只有g改变了。当使用global g 之后,当前函数里所使用的所有对变量g的更改都会对全局变量g进行更改。 除了这个方法,还有引用自己的方法(交互模式里,可以import main)与sys.modules的方法(可以使用引用过的所有模块,交互模式里本身可以用main方式),具体如下。

>>> x = 2  >>> import __main__  >>> __main__.x  2  >>> def glob():  ...     __main__.x += 1  ...  >>> glob()  >>> x  3
>>> x = 2  >>> import  sys  >>> sys.modules['__main__'].x  2  >>> def glob():  ...     sys.modules['__main__'].x += 1  ...  >>> glob()  >>> x  3

四、作用域与嵌套函数

被嵌套函数的作用域也是上级函数里:

>>> def outer():  ...     def inner():  ...             print('inner')  ...

在这里,想调用inner函数,必须是在函数outer里面,不能直接使用。这个时候,我们可以返回内部函数的方法来提取内部函数:

>>> def outer():  ...     def inner():  ...             print('inner')  ...     return inner()  ...  >>> func1 = outer()  inner

工厂函数

工厂函数为:根据要求的对象,一个能够记住嵌套作用域的变量值的函数。这种功能,使用类可以更好的实现,但使用函数也能简单实现。

>>> def funcl(x):  ...     def action(y):  #嵌套函数  ...             return x ** y  #返回x ** y的值  ...     return action  #返回嵌套的函数  ...  >>> a = funcl(3)  #定义x值为3后的嵌套函数赋值  >>> a(2)  9  >>> a(3)  27

可以使用lambda,把之前函数变为:

>>> def func(x):  ...     return lambda y :x ** y  ...  >>> a = func(3)  >>> a(2)  9

关于lamdba的使用可以参考:https://www.cnblogs.com/evening/archive/2012/03/29/2423554.html

循环变量

>>> def makeAction():  ...     acts = []  ...     for i in range(5):  ...             acts.append(lambda x:i ** x)  ...     return acts  ...

执行后,返回值可能不是我们想象的结果:

>>> funclist = makeAction()  >>> funclist[0](2)  16  >>> funclist[4](2)  16

原因在于,i x的时候,i的值取的是for循环结束后的最后一个(4)。我们需要写成lambda x, i=i: i x,这样,lambda里的i是本地变量,跟for循环里的i不冲突,优化后如下:

>>> def makeAction():  ...     acts = []  ...     for i in range(5):  ...             acts.append(lambda x,i = i : i ** x)  ...     return acts  ...  >>> funclist = makeAction()  >>> funclist[0](2)  0  >>> funclist[2](2)  4  >>> funclist[4](2)  16

五、nonlocal语句

nonlocal关键字用来在函数或其他作用域中使用外层(非全局)变量,更多用法可以参考https://blog.csdn.net/youngbit007/article/details/64905070

>>> x = 1  #全局变量x为1  >>> y = 1  #全局变量y为1  >>> def printxy():  ...     x = 2  #printxy里的本地变量x为2  ...     y = 2  #printxy里的本地变量y为2  ...     def setxy():  ...             nonlocal x  #设定x为nonlocal  ...             global y  #设定y为global  ...             x =3  ...             y =3  ...     setxy()  ...     print(x)  ...     print(y)  ...  >>> printxy()  3  #printxy里的x为3  2  #printxy里的y没变,仍为2  >>> x,y  (1, 3)  #全局变量x没变,y变为3