Python基础(五)——闭包与lambda的结合
(1)变量的域
要了解闭包需要先了解变量的域,也就是变量在哪一段“上下文”是有效的(类似局部变量和全局变量的区别),举一个很简单的例子。(例子不重要,就是涉及闭包就要时刻关注这个域)
def test():
msg2 = 'test中的'
print('====',msg1) # ==== 非test中的
msg1 = '非test中的'
test()
print(msg1) # 非test中的
print(msg2) # 报错
(2)什么是闭包
维基百科定义:闭包(Closure)或闭包函数(function Closure),是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。所以有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。
定义可以反复体会,先看一个比较有意思的例子:
def test():
msg = '我是test中的'
def test2():
print(msg)
return test2
g = test()
g() # 我是test中的
这段代码执行到第七行的时候,输出了msg的内容。
我们在对比【(1)变量的域】中的代码例子,那个例子中,同样是第七行的时候是报错,因为很好理解,上一个例子msg2已经脱离了test()函数,也就是局部变量只能在内部使用,不能够全局使用。
然后回到这个例子,这就是闭包存在的意义,我们可以在外部访问局部变量。闭包就是:函数+上下文。注意到我这个例子第五行返回的是test2,是一个函数对象(in Python everything is an object )。这里的 g 就是闭包。而我所说的上下文:就是各种变量环境。所以第七行的return,返回的function不是普通function,是带着上下文环境一起的(也就是test2() 函数中带有msg,而msg其实实在test()中定义的,但是也会被test2()带在身边)。不单单只返回第三第四行两行简单的内容。
(3)闭包疑点
我们再看几个 stockoverflow和 官网上的几个关于闭包的例子与疑点:
1.例子一
adders=[0,1,2,3] for i in [0,1,2,3]:
adders[i]=lambda a: i+a print(adders[1](3)) #
这个里之中adders列表存储了匿名函数,adders[1](3) 就是访问adders[1] 中的匿名函数,参数是3,也就是lambda a:i+a(3传递给a,i 是for...in 循环给的,计算结果是 i + a)。
奇怪的是结果adders[1](3) = 6.我们可能会想应该是4阿,1 + 3 = 4。
我的理解是这样的:因为我们需要注意这里的 i 到底是属于谁的,i 是在 for...in 中定义的,一个循环至始至终只有一个 i ,也就是 i 的引用是不变的,变得是 i 得值,所以lambda中的 i ,adders[0],adders[1]....中的 i 都是指向同一个 i ,而最后 i 是三。adders[0,1,2,3] 中的 lambda 匿名函数的参数 i 全都是同一个,这个 i 因为循环最终值是3. 所以3+3=6.(也就是 i 是什么时候定义的?这个问题考虑好,是在调用lambda之前,也就是for循环开始的时候定义好的)
改进方案:
adders=[0,1,2,3] for i in [0,1,2,3]:
adders[i]=lambda a, b = i: b+a print(adders[0](3)) #
print(adders[1](3)) #
print(adders[2](3)) #
print(adders[3](3)) #
那我们就在lambda之中定义一个b,这个b是记录i,但是adders[ ....] 数组中的 b 是各不相同的引用哦。
2.例子二
squares = []
for x in range(5):
squares.append(lambda: x**2) print(squares[2]()) #
print(squares[3]()) #
情况一模一样。最后全都算 4 *4 = 16。改进如下:
squares = []
for x in range(5):
squares.append(lambda b = x: b**2) print(squares[2]()) #
print(squares[3]()) #
Python基础(五)——闭包与lambda的结合的更多相关文章
- python基础之闭包函数和装饰器
补充:全局变量声明及局部变量引用 python引用变量的顺序: 当前作用域局部变量->外层作用域变量->当前模块中的全局变量->python内置变量 global关键字用来在函数或其 ...
- python基础16_闭包_装饰器
不了解是否其他语言也有类似 python 装饰器这样的东西. 最近才发现ECMAScript6也是有生成器函数的,也有 yield generator 装饰器的基础知识是闭包: # 闭包:嵌套函数, ...
- 【笔记】Python基础五:装饰器
一,什么是装饰器 本质就是函数,功能是为其他函数添加附加功能 原则: 1,不修改被修饰函数的源代码 2,不修改被修饰函数的调用方式 例子: import time def timmer(func): ...
- Python基础(五) python装饰器使用
这是在Python学习小组上介绍的内容,现学现卖.多练习是好的学习方式. 第一步:最简单的函数,准备附加额外功能 # -*- coding:gbk -*- '''示例1: 最简单的函数,表示调用了两次 ...
- python基础五
列表生成式 将列表data=[1,2,3]里的元素都乘2 方法一 data=[1,2,3] for index,i in enumerate(data): data[index] *=2 print( ...
- Python基础三. 函数、lambda、filter、map、reduce
一.概述 函数, 就是用一些语句组织起来实现一组特定的功能, 用来重复调用. 函数的作用及意义:最大化的重用代码和最小化的代码冗余以及对流程的分解. Python中有哪些函数: 内建的函数 第三方模块 ...
- Python基础(7)闭包函数、装饰器
一.闭包函数 闭包函数:1.函数内部定义函数,成为内部函数, 2.改内部函数包含对外部作用域,而不是对全局作用域名字的引用 那么该内部函数成为闭包函数 #最简单的无参闭包函数 def func1() ...
- Python基础(七) 闭包与装饰器
闭包的定义 闭包是嵌套在函数中的函数. 闭包必须是内层函数对外层函数的变量(非全局变量)的引用. 闭包格式: def func(): lst=[] def inner(a): lst.append(a ...
- Python基础总结之认识lambda函数、map函数、filter() 函数。第十二天开始(新手可相互督促)
今天周日,白天在学习,晚上更新一些笔记,希望对大家能更好的理解.学习python~ lambda函数,也就是大家说的匿名函数.它没有具体的名称,也可以叫做一句话函数,我觉得也不过分,大家看下代码,来体 ...
- Python基础编程闭包与装饰器
闭包的定义 闭包是嵌套在函数中的函数. 闭包必须是内层函数对外层函数的变量(非全局变量)的引用. 闭包格式: def func(): lst=[] def inner(a): lst.append(a ...
随机推荐
- Codeforces Round #529 -C- Powers Of Two(二进制拆分)
A positive integer xx is called a power of two if it can be represented as x=2yx=2y, where yy is a n ...
- python大战机器学习——人工神经网络
人工神经网络是有一系列简单的单元相互紧密联系构成的,每个单元有一定数量的实数输入和唯一的实数输出.神经网络的一个重要的用途就是接受和处理传感器产生的复杂的输入并进行自适应性的学习,是一种模式匹配算法, ...
- java实现access数据上传
一. --springMvc实现上传 https://blog.csdn.net/qian_ch/article/details/69258465 --转换成spring64位上传 https://b ...
- NSwag生成客户端调用代码
NetCore2.1 WebAPI 根据swagger.json自动生成客户端代码 https://www.cnblogs.com/hunanzp/p/9297361.html 前言 上一篇博客中我们 ...
- 071 Simplify Path 简化路径
给定一个文档 (Unix-style) 的完全路径,请进行路径简化.例如,path = "/home/", => "/home"path = " ...
- Spring Security在标准登录表单中添加一个额外的字段
概述 在本文中,我们将通过向标准登录表单添加额外字段来实现Spring Security的自定义身份验证方案. 我们将重点关注两种不同的方法,以展示框架的多功能性以及我们可以使用它的灵活方式. 我们的 ...
- 【转载】【MVC 学习 Razor语法】
Razor是MVC3中才有的新的视图引擎.我们知道,在ASP.NET中,ASPX的视图引擎依靠<%和%>来调用C#指令.而MVC3以后有了一套新的使用@标记的Razor语法,使用起来更灵活 ...
- mongodb的投影
mongodb 投影意思是只选择必要的数据而不是选择一个文件的数据的整个.如果一个文档有5个字段,需要显示只有3个,然后选择其中只有3个字段. find() 方法 MongoDB 的find()方法, ...
- dubbo rest返回值异常Incompatible types: declared root type
2018-08-28 17:26:02,208 [http-bio-9090-exec-1][][][][][] ERROR com.wjs.member.plugin.intercepter.Ser ...
- 零基础逆向工程19_PE结构03_代码节空白区添加代码_shellcode
1.获取MessageBox地址,构造ShellCode代码 三种获取方法,参考文章末的一篇帖子. E8 E9计算公式 call 的硬编码:E8 00 00 00 00 jmp 的硬编码:E9 00 ...