前言

前面学了装饰器,那么闭包和装饰器有什么区别呢?
闭包传递的是变量,而装饰器传递的是函数对象,只是传的参数内容不一样,闭包的概念包含了装饰器,可以说装饰器是闭包的一种,它只是传递函数对象的闭包。

先看一个面试题

先看一个经典的面试题,很有代表性, 运行以下代码会输出什么呢?为什么会是这种结果?

def fun():
temp = [lambda x: i*x for i in range(4)]
return temp

for everyLambda in fun():
print(everyLambda(2))

运行结果

6
6
6
6

运行的结果是4个6 ,并不是我们想的 :0, 2, 4, 6。上面的代码用到了列表推导式,还有个匿名函数lambda,直接去阅读不太好理解,可以把匿名函数转成自己定义一个函数.
于是上面的代码等价于:

"""
def fun():
temp = [lambda x: i*x for i in range(4)]
return temp

for everyLambda in fun():
print(everyLambda(2))
"""

def fun():
temp = []
for i in range(4):
def inner(x):
return i*x
temp.append(inner)
return temp

for everyLambda in fun():
print(everyLambda(2))

为了更好的理解,可以先去掉外面的一层fun()

temp = []
for i in range(4):
def inner(x):
return i*x
temp.append(inner)

for everyLambda in temp:
print(everyLambda(2))

这里只定义了一个函数 inner(), 有 2 个变量,i 是函数外部的变量,x 是函数内部的变量。
现在问题的关键在理解函数外部变量和函数内部变量的区别了, 接下来再看一个简单的例子

a = 1

def myfunc(b):
return a+b

print(myfunc(100))
a = 2
a = 3
print(myfunc(100))
print(myfunc(100))

运行结果:101   103   103
也就是函数外部变量a是可变的,后面给a重新赋值了,会替换前面的值。上面的 inner(x) 函数也是一样,外部变量 i 的值是0, 1, 2, 3变化,最后的3 会覆盖前面的值,所以得到的结果都是6
如何解决上面的问题,接下来就是要说的闭包的概念了!

什么是闭包?

闭包就是外部函数中定义了一个内部函数,当外部函数返回内部函数对象(注意是函数对象)时,程序接收了内部函数的定义(此时并未被执行),当再次执行这个返回值时,这个被返回的函数才能被执行。
创建一个闭包必须满足以下几点:

  • 必须有一个内嵌函数

  • 内嵌函数必须引用外部函数中的变量

  • 外部函数的返回值必须是内嵌函数

闭包和装饰器的区别:闭包传递的是变量,而装饰器传递的是函数,除此之外没有任何区别,或者说装饰器是闭包的一种,它只是传递函数的闭包。
以下是闭包的一个标准示例:

def outer(age):
def inner(name):
print("my name is %s. my age is %s." % (name, age))
return inner

demo = outer("18")
demo("yoyo")
# 运行结果:my name is yoyo. my age is 18.

上面的问题,用闭包来解决

# 闭包
def temp(i): # 一个外层函数
def inner(x): # 内层函数
return i*x
return inner

a = []
for i in range(4):
a.append(temp(i))
print(a)

for j in a:
print(j(2))

运行结果

[<function temp.<locals>.inner at 0x000002A8EE929AE8>, 
<function temp.<locals>.inner at 0x000002A8EE929B70>,
<function temp.<locals>.inner at 0x000002A8EE929BF8>,
<function temp.<locals>.inner at 0x000002A8EE929C80>]
0
2
4
6

使用列表推导式

# 闭包
def temp(i): # 一个外层函数
def inner(x): # 内层函数
return i*x
return inner

def fun():
temps = [temp(i) for i in range(4)]
return temps

for everyLambda in fun():
print(everyLambda(2))

这样就可以得到我们的预期结果:0 2 4 6

通过上面的案例就可以了解到闭包的作用了,它保存了函数的外部变量,不会随着变量的改变而改变了。

来源:https://mp.weixin.qq.com/s/vCR8ZzYtu3eUP44Z_Vmwzg

面试题-python 什么是闭包(closure)?的更多相关文章

  1. 说说Python中的闭包 - Closure

    转载自https://segmentfault.com/a/1190000007321972 Python中的闭包不是一个一说就能明白的概念,但是随着你往学习的深入,无论如何你都需要去了解这么一个东西 ...

  2. Python中的闭包 - Closure

    Python中的闭包不是一个一说就能明白的概念,但是随着你往学习的深入,无论如何你都需要去了解这么一个东西. 闭包的概念 我们尝试从概念上去理解一下闭包. 在一些语言中,在函数中可以(嵌套)定义另一个 ...

  3. python 函数对象(函数式编程 lambda、map、filter、reduce)、闭包(closure)

    1.函数对象 作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 秉承着一切皆对象的理念,我们再次回头来看函数(function).函 ...

  4. 【Python】闭包Closure

    原来这就是闭包啊... 还是上次面试,被问只不知掉js里面的闭包 闭包,没听过啊...什么是闭包 回来查了下,原来这货叫闭包啊...... —————————————————————————————— ...

  5. Python深入04 闭包

    作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 闭包(closure)是函数式编程的重要的语法结构.函数式编程是一种编程范式 (而 ...

  6. 21.python中的闭包和装饰器

    python中的闭包从表现形式上定义(解释)为:如果在一个内部函数里,对在外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认为是闭包(closure). 以下说明主要针对 python ...

  7. Swift语言精要-闭包(Closure)

    闭包(Closure)这个概念如果没学过Swift的人应该也不会陌生. 学过Javascript的朋友应该知道,在Javascript中我们经常会讨论闭包,很多前端工程师的面试题也会问到什么是闭包. ...

  8. Python 中的闭包与装饰器

    闭包(closure)是函数式编程的重要的语法结构.闭包也是一种组织代码的结构,它同样提高了代码的可重复使用性. 如果在一个内嵌函数里,对在外部函数内(但不是在全局作用域)的变量进行引用,那么内嵌函数 ...

  9. 【转】python中的闭包

    转自:http://www.cnblogs.com/ma6174/archive/2013/04/15/3022548.html python中的闭包 什么是闭包? 简单说,闭包就是根据不同的配置信息 ...

  10. Python 入门之 闭包

    Python 入门之 闭包 1.闭包 (1)在嵌套函数内使用(非本层变量)和非全局变量就是闭包 (2)_ closure _ 判断是不是闭包 def func(): a = 1 def foo(): ...

随机推荐

  1. Django - 在后台上传文章封面图 - 并在前端页面展示

    需要用到 models.ImageField(), 它继承自 models.FileField(),  用ImageField的时候需要安装pillow pip install pillow -i h ...

  2. 一款功能强大的Python工具,一键打包神器,一次编写、多平台运行!

    1.项目介绍 Briefcase是一个功能强大的工具,主要用于将Python项目转化为多种平台的独立本地应用.它支持多种安装格式,使得Python项目能够轻松打包并部署到不同的操作系统和设备上,如ma ...

  3. zabbix笔记_003 配置微信告警

    配置邮件告警 安装python-requests,使用微信发送告警 发送告警报错: yum install -y python-requests 测试告警: cat weixin.py #------ ...

  4. MyBatis实现MySQL表字段及结构的自动增删

    前言 在开发过程中,总会涉及到数据库表结构字段的增加或者删除,或者是索引的增加和减少,这个时候能把修改表结构字段这些工作都交给程序来进行,那能大大方便开发.正好有一个现成的工具可以在springboo ...

  5. 使用nvm安装以及管理多版本node教程

    安装nvm.node.npm 下载nvm安装包,推荐使用1.1.7,我个人使用1.1.8会有中文乱码的报错 点击exe文件,注意修改nvm的安装根目录以及node的安装根目录,后者是以后管理多版本no ...

  6. 三大显卡厂商(Intel NVIDIA AMD)产品对硬件解码编码支持程度列表

    三大显卡厂商(Intel NVIDIA AMD)产品对硬件解码编码支持程度列表 以下全部为重庆Debug原创博客园独发,资料收集不易,贵请珍惜,更新日期2023年6月8日 Intel GPU解码编码的 ...

  7. 2024-06-08:用go语言,给定三个正整数 n、x和y, 表示城市中的房屋数量以及编号为x和y的两个特殊房屋。 在这座城市中,房屋通过街道相连。对于每个编号i(1 <= i < n), 存在一条

    2024-06-08:用go语言,给定三个正整数 n.x和y, 表示城市中的房屋数量以及编号为x和y的两个特殊房屋. 在这座城市中,房屋通过街道相连.对于每个编号i(1 <= i < n) ...

  8. WIN8 WIN10 WIN11离线安装 .NET 3.5

    WIN8 WIN10 WIN11离线安装 .NET 3.5 以WIN10为例: 1.双击WIN10 ISO 镜像,看挂载的是哪个盘符.我这边是E:. 2.使用 WIN + X 快捷键,打开 Windo ...

  9. 内部网关协议RIP-路由选择协议

    路由信息协议RIP(Routing Information Protocol)是内部网关协议IGP中最先得到广泛使用的协议,其相关标准文档为RFC1058. 一.RIP基本工作原理 RIP要求自治系统 ...

  10. vite+vue3+ts+elementPlus前端框架搭建 [三] router路由管理

    路由包括动态路由.静态路由两种,本文中以静态路由的方式实现了动态路由. 1. 创建Router 在Src目录下创建router文件夹,并在router文件夹下创建index.ts文件. index.t ...