前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【python入门到精通】python高级知识点之闭包的使用

【python入门到精通】python高级知识点之闭包的使用

作者头像
大数据小禅
发布2021-12-21 14:49:44
2340
发布2021-12-21 14:49:44
举报
文章被收录于专栏:YO大数据

作者 :“大数据小禅”

目录

高级知识点之闭包

在万物皆对象的Python中,函数是否能作为函数的返回值进行返回呢?

代码语言:javascript
复制
def my_power():
   n = 2
   def power(x):
     return x ** n
   return power    #直接把这个函数给return回去

p = my_power()
print(p(4))

输出:
16
代码语言:javascript
复制
def my_power():
  n = 2
  def power(x):
    return x ** n
  return power

n = 3
p = my_power()   #在调用函数的时候n=2就被引用,不受n=3的影响。
print(p(4))

输出:
16

我们可以看到,my_power函数在返回的时候,也将其引用的值(n)一同带回,n的值被新的函数 所使用,这种情况我们称之为闭包

当我们把n的值移除到my_power函数外面,这个时候来看下计算结果

代码语言:javascript
复制
n = 2
def my_power():
  def power(x):
     return x ** n
  return power
n = 3                #虽然上面由一个n=2,但是会找到距离最近的一个n=3替换进来。
p = my_power()
print(p(4))

输出:
64
为什么输出的结果会是64?

我们先来看看闭包时,p.__closure____的结果

代码语言:javascript
复制
例1
def my_power():
  n = 2
  def power(x):
    return x ** n
  return power
p = my_power()
print(p.__closure__)
结果:(<cell at 0x00000264518F9A38: int object at 0x00007FFA7F617120>,)  #int类型
closure是内部函数的一个属性,用来保存环境变量
代码语言:javascript
复制
例2
n = 2
def my_power():
  def power(x):
   return x ** n
  return power
n = 3
p = my_power()
print(p.__closure__)
输出结果 None 

通过例1跟例2对比,我们可以知道,例2并不是闭包。注意了:!!!闭包指的是,在返回一个函数的同时,将这个函数所携带的一些环境,比如说环境变量n=2,去进行这个返回,而类似第二种就不是一个闭包,他返回的是一个函数。!!

高频面试点之闭包经典问题

下面的程序是否是闭包?能否正确运行

代码语言:javascript
复制
def my_power():
   n = 2
   def power(x):
     n += 1         #当在闭包里面的时候,没有办法直接修改n=2的值,所以这个程序会报错
     return x ** n
   return power
p = my_power()
print(p(3))

如何让上面的程序正确运行?看看改正之后的结果

代码语言:javascript
复制
def my_power():
   n = 2
   def power(x):
     nonlocal n         #表示引用这个函数外面非全局变量,进行使用,这时候就可以修改这个值。
     n += 1
     return x ** n
   return power

p = my_power()
print(p.__closure__)    
#打印出来的是(,)可以看到这个变量类型被携带出来。
print(p(3))
print(p(3))

输出:
(<cell at 0x000001BD4D1BC738: int object at 0x00007FFEB90B7120>,)
27
81
看看下面的程序的运行结果
代码语言:javascript
复制
def my_power():
  n = 2
  L = []
  for i in range(1, 3):    #因为这里产生1,2两个数字,所以下面会循环两次,要用f1,f2两个变量值来接收,不然报错
   def power():
     return i ** n       #这里的i值是不可以确定的,在python中如果连传入的形参都不确定的话,那么python编译完会默认记住上面那个for循环的最后一个值,也就是都是2,所以导致最后面输出两个都是4,而不是一个1,一个4.
   L.append(power)
  return L

f1, f2 = my_power()
print(f1())
print(f2())

print(f1.__closure__[0].cell_contents)   #看闭包里面携带的环境变量是多少
print(f2.__closure__[0].cell_contents)

输出:
4
4
2
2

python的函数只有在执行时,才会去找函数体里的变量的值,也就是说你连形参都不确定,你咋求知道 i为几呢?,在这里,你只需要记住如果你连形参都不确定,python就只会记住最后一个i值。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2021/12/20 ,如有侵权请联系 cloudcommunity@tencent.com 删除

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

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

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

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 目录
  • 高级知识点之闭包
  • 高频面试点之闭包经典问题
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档