面试题-python 什么是闭包(closure)?

前言

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

先看一个面试题

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

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

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

2021年第六期《python接口自动化+测试开发》课程,1月9号开学(火热报名中!)

本期上课时间:1月9号-4月18号,每周六、周日晚上20:30-22:30

(0)

相关推荐

  • Python学习—装饰器

    学习Python已经有一段时间了,陆续学了一些基础部分,但是理解的不是很深刻,每过一段时间就会忘记,所以不得不写一些博客进行记录,加深自己的理解.这两个星期一直在研究装饰器,开始觉得很简单,但是只知其 ...

  • 一文看懂Python系列之装饰器(decorator)(工作面试必读)

    Python的装饰器(decorator)可以说是Python的一个神器,它可以在不改变一个函数代码和调用方式的情况下给函数添加新的功能.Python的装饰器同时也是Python学习从入门到精通过程中 ...

  • 深入理解 Python 内部函数和闭包(进阶)

    大家好,我是安果! 本文以内部函数为主线,深入讲解内部函数和闭包的应用场景和原理,学会后你的 Python 水平会再上一个台阶,对工作面试或实战应用都会很有帮助 本文包括: 函数是一等公民 内部函数定 ...

  • python - 内置对象 之 变量

    一.变量命名规划 1.命名规则 (1)命名内容只能是字母.下划线.数字 (2)名字第1字符只能是字母或下划线 (3)区分大小写 2.私有变量 (1)_xxx "单下划线" 开始的成 ...

  • 初识装饰器函数

    我之前看装饰器文章介绍,很少有用 装饰器函数这种称谓的.但是今天大邓简单的学了下装饰器,觉得应该先让大家知道装饰器是一种函数,让大家从熟悉的函数去学习装饰器.大邓姑且草率的将其称呼为 装饰器函数 装饰 ...

  • 函数与Lambda表达式

    函数参数定义 # 学习人员:贾其豪# 开发时间:2021/1/31 14:05#函数定义默认值参数#函数定义时,给形参设置默认值,只有与默认值不符的时候才需要传递实参def fun(a,b=10): ...

  • 面试题-python 浅拷贝和深拷贝(copy模块)

    前言 面试的时候经常会问到深拷贝和浅拷贝,那么python的深拷贝和浅拷贝有什么区别呢? 思考题 先来看 2 个简单的案例, 对元素 a/aa 重新赋值一个新的变量 b/bb 后,改变原来 a/aa ...

  • 面试题-python 什么是迭代器?

    前言 python 里面有 3 大神器:迭代器,生成器,装饰器.在了解迭代器之前,需弄清楚2个概念: 1.什么是迭代 2.什么是可迭代对象 迭代 如果给定一个list或tuple,我们可以通过for循 ...

  • 面试题-python 什么是生成器(generator)?

    前言 在 Python 中,带有 yield 的函数在 Python 中被称之为 generator(生成器). 跟普通函数不同的是,生成器是一个返回迭代器的函数,只能用于迭代操作,更简单点理解生成器 ...

  • 面试题-python 如何读取一个大于 10G 的txt文件?

    前言 用python 读取一个大于10G 的文件,自己电脑只有8G内存,一运行就报内存溢出:MemoryError python 如何用open函数读取大文件呢? 读取大文件 首先可以自己先制作一个大 ...

  • 面试题-python 垃圾回收机制?

    前言 简历上写着熟悉 python 面试官上来就问:说下python 垃圾回收机制?一盆冷水泼过来,瞬间感觉 python 不香了. Python中,主要通过引用计数(Reference Counti ...

  • 前端面试题整理——作用域和闭包

    什么是闭包,闭包的表现形式: // 作用域应用的特殊情况,有两种表现: // 函数作为参数被传递 // 函数作为返回值被返回 // 函数作为返回值 function create() { let a ...

  • Python 中的函数装饰器和闭包

    函数装饰器可以被用于增强方法的某些行为,如果想自己实现装饰器,则必须了解闭包的概念. 装饰器的基本概念 装饰器是一个可调用对象,它的参数是另一个函数,称为被装饰函数.装饰器可以修改这个函数再将其返回, ...

  • 【必看】Python面试题之框架与前端!

    学完Python后,接下来的重点就是准备面试,而在面试过程中,面试官总会问到一些技术性的问题;对此,为让大家面试更加顺利,做好充足准备,小编为大家总结了150道有关Python前端和框架的经典面试题, ...