python的全局变量和局部变量

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

x = "global x"

def level_one():
    return x

print(level_one())

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

x = "global x"

def level_two(control):
    if control:
        x = "local x"
    return x

print(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 inner

f = 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 x
overwritten global
nonlocal x
overwritten global

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

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值