文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Python编程中闭包的变量作用域问题解析

2024-04-02 19:55

关注

闭包

​ 在我们使用返回函数的时候,由于我们在一个函数中需要返回另一个函数,因此,我们在这个函数中就需要重新定义一个函数。而这样,就造成了我们的函数嵌套问题。外面的函数相对于里面的函数而言是外函数(outer function),而里面的我们叫他内函数(inner function)。


def outerFunction(): #外函数
    def innerFunction(): #内函数
        x = 1
        return x
    return innerFunction #返回值是一个函数
a = outerFunction()
print(a)

这里我们打印 a 的值得时候,实际上打印的是我们的返回函数的地址 :

<function outerFunction.<locals>.innerFunction at 0x0000019C278C0E50>

​ 一般情况下,我们在使用函数作为返回值得时候,我们会在内函数中使用我们外函数中的变量,这种情况就会产生一个有意思的事情了。内函数会被返回给外部的其他调用者,而我们的变量是在我们的外函数中定义的。此时,我们的变量的作用域会发生怎样的变化呢?

测试:


def outerFunction(x): #外函数
    y = 10
    def innerFunction(): #内函数
        return x + y
    return innerFunction #返回值是一个函数
a = outerFunction(10)
print(a())

​打印:

20

​ 这里可以看到,我们的在给 a 赋值的时候,同时给外函数传进去了一个值10,然后,我们直接把 a() 打印出来,此时,我们的 a 返回了 20,说明我们的变量和参数在进入内函数后,我们的内函数会保留这个变量的值。

​ 这里,我们把这种现象叫做“闭包(Closure)”,我试着多次返回这个内函数,看看他们的地址是否一致:


def outerFunction(x): #外函数
    y = 10
    def innerFunction(): #内函数
        return x + y
    return innerFunction #返回值是一个函数
a = outerFunction(10)
b = outerFunction(20)
c = outerFunction(30)
print(a())
print(b())
print(c())
print(a)
print(b)
print(c)

​ 打印:

20
30
40
<function outerFunction.<locals>.innerFunction at 0x0000020C480C0DC0>
<function outerFunction.<locals>.innerFunction at 0x0000020C480C0D30>
<function outerFunction.<locals>.innerFunction at 0x0000020C480CD280>

​ 这里我们可以看到每个 innerFunction 的地址都不同,当我们多次调用返回函数的时候,每调用一次,我们的返回函数就会新创建一个函数给我们的变量。

​ 那么,如果我们在外函数里面多次调用我们的内函数,我们的外函数的变量的作用域是什么样的呢?

​ 我们测试一下:


def outerFunction(x): #外函数
    L=[] #定义一个List 
    y = 10
    for i in range(1, 4):
        def innerFunction(): #内函数
            return (x + y) * i #使用我们外函数得变量
        print(innerFunction()) #打印内函数返回得值
        #print(innerFunction)
        L.append(innerFunction) #把内函数添加到我们得List中
    return L #返回这个List
a = outerFunction(10)
print(a[0])
print(a[1])
print(a[2])
print(a[0]())
print(a[1]())
print(a[2]())

​ 看打印:

20
40
60
<function outerFunction.<locals>.innerFunction at 0x00000274AD6B0E50>
<function outerFunction.<locals>.innerFunction at 0x00000274AD6BD040>
<function outerFunction.<locals>.innerFunction at 0x00000274AD6BD3A0>
60
60
60

​ 这里我们可以看到一个有意思得现象:

​ 当我们在我们的函数内多次调用我们的内函数,并把它的返回值打印出来的时候,我们可以看到这个值是正常的(依次递增)。而如果我们把这个内函数传递到我们的 List 中,在函数外部调用它的时候,我们的函数的返回值全部变成了 60 也就是他们的只得到了我们外函数中变量的最终的值。这个就是我们的闭包特有的现象。

闭包中的变量

​ 一般情况下,当一个函数结束的时候,在内存中,我们这个函数内部的局部变量会连同这个函数一起被释放掉。

​ 但是,当我们这个函数包含闭包的时候,也就是说,当这个函数的返回值是一个函数的引用的时候。此时,当这个函数结束时,由于它内部的局部变量需要被释放掉,因此,它会把这个变量的值给传递给它所要返回的这个函数的内部。正是因为这样,当我们在外部调用我们的返回函数的时候,我们会看到它使用的变量全都是这个变量在函数内的最终的值,因为这个变量是这个外函数在结束的时候才传递给我们的返回函数的。而此时,我们函数内的变量只能有一个值。

​ 但是,我们可以使用另一个办法来规避这种情况:


def outerFunction(x): #外函数
    L=[] #定义一个List 
    y = 10
    def innerFunction(i): #内函数
        def f():
            return (x + y) * i #使用我们外函数得变量
        return f
    for i in range(1, 4):
        print(innerFunction(i)()) #打印内函数中的内函数的返回值
        L.append(innerFunction(i)) #把内函数添加到我们得List中
    return L #返回这个List
a = outerFunction(10)
print(a[0])
print(a[1])
print(a[2])
print(a[0]())
print(a[1]())
print(a[2]())

​ 打印如下:

20
40
60
<function outerFunction.<locals>.innerFunction.<locals>.f at 0x00000155B0F80040>
<function outerFunction.<locals>.innerFunction.<locals>.f at 0x00000155B0F803A0>
<function outerFunction.<locals>.innerFunction.<locals>.f at 0x00000155B0F80430>
20
40
60

​ 这里我们可以看到我们在外部调用的时候,函数的返回值和在内部调用的返回值是一样的。那么我们分析一下这个函数的执行过程以及函数内的变量的作用域情况。

​ 首先,我们在我们由于的内函数的基础又添加了一层内函数 f() ,并且,在这个内函数里面,我们使用了外部的内函数 innerFunction 的参数作为了返回值。然后,我们在 for 循环内部把我们的 innerFunction 这个函数的返回值添加进来我们的 List 里面。

​ 这里我们应该已经发现了。在 for 循环内部,当我们把 innerFunction 的返回值 f 添加到我们的 List 中的时候,由于 innerFunction 相对于函数 f() 而言,f() 是属于 innerFunction 的内函数的,因此,当我们返回 f() 这个函数的时候,此时,f() 函数内部使用的值就是它的最终值了,而此时,我们的 innerFunction 函数还在 for 循环内部,因此,在循环内部,每次调用我们的 innerFunction 的返回值时,我们的传递到我们的 f() 函数中的值就是我们的最终值了。

​ 此时,即使是我们在外部调用这个 f() 函数,它返回的值和内部调用时也是一样的。

以上就是Python编程中闭包的变量作用域问题解析的详细内容,更多关于Python闭包中变量作用域的资料请关注编程网其它相关文章!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯