python闭包

关于闭包, 很多blog中都这样解释 :对于一个嵌套定义的函数,外层的函数的返回值是内层函数,而在内层函数中又引用了外层函数的局部变量,在外层函数执行后,其局部变量并非被回收,而会同返回的内层函数一同存在,而这一现象被称为闭包(closure)。

不过以上的理解有些繁琐和局限, 在计算机科学中 ,闭包(Closure)词法闭包(Lexical Closure)的简称,是引用了自由变量的函数。 这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。 也即对于第一段中的定义可以适当放开一些限制条件,python中的闭包实现也并非那么局限。

引用

通过上文介绍可以对于python闭包有大概的了解, 但是有些看似简单的细节却需要进一步阐述 。

python中变量的概念,这是与C/C++中极为不同的,在C/C++中变量是一个名称与内存合一的实体,改变一个变量的值,并不改变其内存的地址。 而变量这个概念在python中并不合用,很多场合它的运用都会让人混淆 。

python中所使用的概念是引用和对象,即如a=123,a即是一个引用名称,123是内存中所储存的对象值。这其实更像是C/C++中的指针与其所指向的内存,可以看作python在此之上对语法进行了包装。

回到之前讨论的闭包话题,在其中用到了 变量 的概念,即函数引用的 变量 将与函数一同存在,这里的 变量 其实是引用名称与内存对象的复合概念。我们这里对其进行进一步的阐明:

函数中所使用的外层函数引用名称(指针),在外层函数退出后其所指向的内存对象并不回收,而该引用名称(指针)会与内层函数一同存在,虽然此时该引用名称(指针)对于内层函数不是“可见的”。

陷阱

def count(): 
  fs = [] 
  for i in range(1, 4): 
    def f(): 
      return j*j 
    fs.append(f)
  return fs

f1, f2, f3 = count()
print(f1())
print(f2())
print(f3())

对于以上代码,假如按照C/C++中的概念去理解python中的变量,就会以为其输出依次为1、2、3。其实不然,真正输出为:3、3、3。根据上一小节中对于python中引用与闭包的阐述,在内存f函数中使用外层的引用名称i,在循环中虽然将不同的f函数加入到列表fs中,但是它们都使用的是同一个引用i,而该引用最后对应的值为3。

再看一段代码,这个会稍微复杂一点

def test():
  for i in range(4):
    yield i
    
g=test()

for n in [1,10]:
  g=(n+i for i in g)
  
print(list(g))

上面这段代码的输出,一时不查之下也会以为是11、12、13、14,而其真实结果却是20、21、22、23,让人一时抓不到头脑。首先在for循环中的生成器表达式(n+i for i in g),它其实本质上是一个函数,写成表达式的形式不过是一种语法糖,其函数形式为:

def gen(n):
  # g是外面全局的那个生成器g
  for i in g:
    yield n+i

即生成器generator本身是一种算法或是函数,只有在“调用”它的时候,也就是对其进行for或是list或是next之类的操作时,才会真正的有值流动。

那么对于以上第二例子中的代码,在for循环内n=1时,g这个生成器被重新赋值,但注意它此时只是一个特殊的函数,此时的n与i并没有真正相加,在for循环的第二轮n=10的时候,(n+i for i in g)表达式中对g才进行了调用,那么此时流进函数的n值其实是10,也就是此时g这个生成器对应的值为10、11、12、13,也就是i所引用的是这些值,下面又以相同的n+i的形式创造一个新的生成器对g重新赋值,并退出循环。则自然,此时g中对应的值为20、21、22、23.

以上就是python闭包与引用以及需要注意的陷阱的详细内容,更多关于python 闭包与引用的资料请关注其它相关文章!

标签:
python,闭包,python,引用

免责声明:本站文章均来自网站采集或用户投稿,网站不提供任何软件下载或自行开发的软件! 如有用户或公司发现本站内容信息存在侵权行为,请邮件告知! 858582#qq.com
评论“python闭包与引用以及需要注意的陷阱”
暂无“python闭包与引用以及需要注意的陷阱”评论...

《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线

暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。

艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。

《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。