龙空技术网

python的全局变量和局部变量

搬砖搬知识 584

前言:

眼前同学们对“python中的全局变量可以直接修改”都比较注重,看官们都想要学习一些“python中的全局变量可以直接修改”的相关知识。那么小编在网络上收集了一些关于“python中的全局变量可以直接修改””的相关知识,希望各位老铁们能喜欢,小伙伴们快快来学习一下吧!

前言

对于初学python的小伙伴,觉得自己对python的全局变量和局部变量很了解,但是你真的了解吗?或许你可以看一下以下内容。在看代码后,先不要急着去看运行的结果,先自己想一下结果该是什么。

代码

x = "global x"def level_one():    return xprint(level_one())

这段代码应该没有什么疑问,x取值的是全局变量x。加点难度,看一下下面的代码。

x = "global x"def level_two(control):    if control:        x = "local x"    return xprint(level_two(True))print(level_two(False))

这段代码第一次传入True很好理解,输出的是局部变量的x值。但是第二次传入False就要小心了,它会报错(UnboundLocalError: local variable ‘x’ referenced before assignment),原因是虽然control是False,后面的语句不会运行到,但是对于python而已,即使没有运行到x的赋值,return x的时候,会把x当初局部变量,而这时的局部变量x没有进行赋值操作。解决办法是在return之前对x做一个赋值操作。

x = "global x"def level_three():    def inner(y):        return x, y, z        z = "outer z"    return inner("y arg")print(level_three())

这段程序最大的疑点是z值,如果语句(z = “outer z”)在语句(def inner(y):)之前那估计你没有什么疑问,但是在之后你可能会觉得程序会报错。其实程序能正确的输出(‘global x’, ‘y arg’, ‘outer z’),z能正确的转入inner函数,得益于__closure__的机制。

x = "global x"def level_four():    def inner(y):        return x, y, z    print(inner.__closure__)    z = "first outer z"    print(inner.__closure__)        z = "second outer z"    print(inner.__closure__)    print(inner.__globals__)    return inner("y arg")print(level_four())

我们把inner的__globals__和__closure__参数打出来代码运行结果如下:

(<cell at 0x7fbbd9190520: empty>,)(<cell at 0x7fbbd9190520: str object at 0x7fbbd91f5f70>,)(<cell at 0x7fbbd9190520: str object at 0x7fbbd91f6030>,)°'__name__': '__main__', '__doc__': None, '__package__': None, '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x7fbbd9128d90>, '__spec__': None, '__annotations__': °ç, '__builtins__': <module 'builtins' (built-in)>, '__file__': '/home/shouqi/workspace/python/main.py', '__cached__': None, 'x': 'global x', 'level_four': <function level_four at 0x7fbbda31bd90>ç('global x', 'y arg', 'second outer z')

从上面可以看到,x是全局变量,由于没有局部变量x,所以使用的是全局变量x。y的值应该也没有异议,毕竟是传入的。对于z值其实也没有异议,这里打出值来,就是想让大家看看,z值没赋值之前是empty,等两次赋值,值是基于最后一次赋值。

x = "global x"def level_five(n):    z = f"outer z {n}"    def inner(y):        return x, y, z        return innerf = level_five(0)g = level_five(1)print(f("y arg"), g("other y arg"))

这样的函数调用很奇怪,变量f和g类似于函数变量,在后面的print语句中进行调用。输出的结果是(‘global x’, ‘y arg’, ‘outer z 0’) (‘global x’, ‘other y arg’, ‘outer z 1’)。

x = "global x"def level_six():    z = "outer z"    def donky():        def inner(y):            return x, y, z                z = "donky z"        return inner        def chonky():        x = "chonky x"        f = donky()        return f("y arg")        return chonky()print(level_six())

你先想一下输出的结果,这里可能最大的争议是x的值。其实chonky函数中的局部变量x不会通过函数传入,所以输出的x值是全局变量的值。结果:(‘global x’, ‘y arg’, ‘donky z’)。

x = "global x"def nonlocal_and_global():    x = "nonlocal x"    def f():        global x        x = "overwritten global"        return x        print(x)    print(f())    print(x)nonlocal_and_global()print(x)

这段代码是用到了global关键字,所以可能迷惑的地方是第二次打印x的值和最后一次打印x的值。但整体仔细一点应该还是不会错的,结果:

nonlocal xoverwritten globalnonlocal xoverwritten global

结束语

以上是python的一些常见的局部变量和全局变量的问题。

标签: #python中的全局变量可以直接修改