Python 嵌套函数和闭包

1、函数嵌套

如果在一个函数内部定义了另一个函数,我们称外部的函数为外函数,内部的函数为内函数,如下代码:

def out_func():
def inner_func1(): # 在内部定义函数1
print("the first inner func")
return def inner_func2(): # 在内部定义函数2
print("the second inner func")
return inner_func1() # 调用内部函数1
inner_func2() # 调用内部函数2 return out_func() # 调用外部函数

运行输出结果为:

the first inner func
the second inner func

在out_func()这个函数内部,定义了inner_func1()和inner_func2,随后也调用了这两个函数。其实完全可以把这两个函数写在out_func()的外面,如下,其执行结果和上面嵌套函数一样:

def inner_func1():
print("the first inner func")
return def inner_func2():
print("the second inner func")
return def out_func():
inner_func1()
inner_func2()
return out_func() # 调用out_func函数

2、闭包

在一个外函数内定义了一个内函数,内函数里运用了外函数的临时变量,并且外函数的返回值是内函数的引用,这样就构成了一个闭包。

一般情况下,如果一个函数结束,函数内部所有的东西都会释放掉,还给内存,局部变量都会消失。但是闭包是一种特殊情况,如果外函数在结束的时候发现有自己的临时变量将来会在内部函数中用到,就把这个临时变量绑定给了内部函数,然后自己再结束。

# 闭包函数的实例:outer()是外部函数,a和b都是外部函数的临时变量
def outer(a):
b = 10
# inner()是内部函数 def inner():
# 在内部函数中用到了外部函数的临时变量
print(a+b) # 外部函数返回值是内部函数的引用
return inner # 调用外部函数,传入参数5。此时外函数两个临时变量a是5 b是10,并创建了内部函数,然后把内部函数的引用返回给了demo进行存储,外部函数结束的时候发现内部函数将会调用到自己的临时变量,这两个临时变量就不会释放,会绑定给这个内部函数
demo1 = outer(5) # demo存储了外部函数的返回值,也就是inner()函数的引用,这里相当于执行inner函数
demo1() # 15 demo2 = outer(7)
demo2() # 17

说明:

  • 外函数返回内函数的引用:在Python中一切都是对象,当我们进行变量赋值时(a=1),实际上在内存中已经存储了值1,然后用a这个变量名存储1所在内存位置的引用。引用就好比C语言的指针,可以理解为地址。a只是变量名,a里面存储的是1的这个值所在的地址,就是a里面存储了数值1的引用。同理,当我们在Python中顶一个函数def demo():时,内存中会开辟一些空间,存下这个函数的代码、内部的局部变量等等。这个demo只不过是一个变量名,它里面存了这个函数所在位置的引用而已。我们还可以进行x=demo,y=demo,这样的操作就相当于把demo里存的内容赋值给x和y,这样x和y都指向了demo函数所在的引用,在这之后我们可以用x()或者y()来调用我们创建的demo(),调用的本质就是执行一个函数,x、y和demo三个变量名存了同一个函数的引用。对于上面的例子,在外部函数outer中最后返回inner,我们在调用外部函数demo=outer()时,outer返回了inner,inner是内部函数的引用,这个引用被存入了demo中,所以接下来我们再进行demo()时,相当于执行了inner函数。
  • 在内部函数中想修改闭包变量(外部函数绑定给内部函数的局部变量)时:Python3中,可以使用nonlocal关键字声明一个变量,表示这个变量不是局部变量空间的变量,需要向上一层变量空间找这个变量;Python2中没有nonlocal这个关键字,可以把闭包变量改成可变类型数据进行修改,比如:列表。
# 修改闭包变量实例
def outer(a): # outer是外部函数,a和b都是外部函数的临时变量
b = 10 # a和b都是闭包变量
c = [a] # 这里对应修改闭包变量的方法2 def inner(): # inner是内部函数
# 内函数中想修改闭包变量
nonlocal b # 方法1 nonlocal关键字声明
b += 1 c[0] += 1 # 方法2 把闭包变量修改成可变数据类型,比如:列表
print(c[0], b) return inner # 外部函数返回内部函数的引用 demo = outer(5)
demo() # 6 11

注意:使用闭包的过程中,一旦外部函数被调用一次返回了内部函数的引用,虽然每次调用内部函数是开启一个函数执行过后消亡,但是闭包变量实际上只有一份,每次开启内部函数都在使用同一份闭包变量,如下代码:

def outer(x):
def inner(y):
nonlocal x
x += y
return x
return inner demo = outer(10)
print(demo(1), demo(3)) # 11 14

由此可见,每次调用inner的时候,使用的闭包变量x实际上是同一个。

闭包的用途:

  • 装饰器:详细参考Python 装饰器
  • 面向对象:在Python中虽然我们不这样使用,但是在其他编程语言,比如:JavaScript中,经常用闭包来实现面向对象编程;
  • 单利模式:其实这也是装饰器的应用,关于设计模式后面再讨论;

Python 嵌套函数和闭包的更多相关文章

  1. python嵌套函数、闭包与decorator

    1 一段代码的执行结果不光取决与代码中的符号,更多地是取决于代码中符号的意义,而运行时的意义是由名字空间决定的.名字空间是在运行时由python虚拟机动态维护的,但是有时候我们希望能将名字空间静态化. ...

  2. 一文搞懂Python函数(匿名函数、嵌套函数、闭包、装饰器)!

    Python函数定义.匿名函数.嵌套函数.闭包.装饰器 目录 Python函数定义.匿名函数.嵌套函数.闭包.装饰器 函数核心理解 1. 函数定义 2. 嵌套函数 2.1 作用 2.2 函数变量作用域 ...

  3. python基础 (函数名,闭包,和迭代器)

    1.函数名作用 函数名本质上就是函数的内存地址或对象. 1.可以被引用 2.可以被当作容器类型的元素 3.可以当作函数的参数和返回值 4.如果记不住的话,那就记住一句话,就当普通变量用 2.闭包 什么 ...

  4. python之函数名,闭包、迭代器

    一.函数名的运用(第一类对象) 函数名是一个变量,但它是一个特殊的变量,与括号配合可以执行函数的变量. 1,函数名的内存地址: def func(): print("呵呵") pr ...

  5. python函数作用域,嵌套函数,闭包

    函数作用域                                                                                                ...

  6. python函数调用顺序、高阶函数、嵌套函数、闭包详解

    一:函数调用顺序:其他高级语言类似,Python 不允许在函数未声明之前,对其进行引用或者调用错误示范: def foo(): print 'in the foo' bar() foo() 报错: i ...

  7. Python返回函数、闭包,匿名函数

    函数不仅可以作为函数参数,还可以作为函数返回结果 def pro1(c,f): def pro2(): return f(c) return pro2 #调用pro1函数时,返回的是pro2函数对象& ...

  8. Python之函数进阶

    本节内容 上一篇中介绍了Python中函数的定义.函数的调用.函数的参数以及变量的作用域等内容,现在来说下函数的一些高级特性: 递归函数 嵌套函数与闭包 匿名函数 高阶函数 内置函数 总结 一.递归函 ...

  9. 【转】Python之函数进阶

    [转]Python之函数进阶 本节内容 上一篇中介绍了Python中函数的定义.函数的调用.函数的参数以及变量的作用域等内容,现在来说下函数的一些高级特性: 递归函数 嵌套函数与闭包 匿名函数 高阶函 ...

随机推荐

  1. object转List<XXX>的问题

    List<object> demo(object a) { List<object> res = new List<object>(); object c = a; ...

  2. libevent学习四(Working with events)

    1.事件的分类 文件可写 文件可读 超时发生 信号发生 用户触发事件   2事件的生命周期        --非 persistent                                 ...

  3. 「知识学习」二分图的最大匹配、完美匹配和匈牙利算法(HDU-2063)

    定义 如果一个图\((E,V)\)的顶点集\(E\)能够被能够被分成两个不相交的集合\(X,Y\),且每一条边都恰连接\(X,Y\)中的各一个顶点,那么这个图就是一个二分图. 容易得知,它就是不含有奇 ...

  4. Selenium(Python) ddt读取MySQL数据驱动

    import unittestfrom time import sleep from ddt import ddt, datafrom pymysql import connectfrom selen ...

  5. Matlab R2016a 破解教程

    郑重声明:图片来源于网络,在这里感谢图片提供者,我写这篇教程,是希望帮助后来者少走弯路,而且,这是一种比较简单有效的破解方法,针对网上那种修改本地文件的方法,在这里不做介绍,如果想体验,可自己百度或谷 ...

  6. Python简要标准库(2)

    集合 堆 和 双端队列 1.集合 创建集合 s = set(range(10)) 和字典一样,集合元素的顺序是随意的,因此不能以元素的顺序作为依据编程 集合支持的运算 a = set([1,2,3]) ...

  7. 8月leetcode刷题总结

    刷题链接:https://leetcode-cn.com/explore/ 根据leetcode的探索栏目,八月份一直在上面进行刷题.发现算法题真的好难,真-计算机思维. 核心是将现实问题转化为计算机 ...

  8. vector的基础使用

    vector是一个容器,实现动态数组. 相似点:下标从0开始. 不同点:vector创建对象后,容器大小会随着元素的增多或减少而变化. 基础操作: 1.使用vector需要添加头文件,#include ...

  9. 204. Singleton

    Description Singleton is a most widely used design pattern. If a class has and only has one instance ...

  10. 孤荷凌寒自学python第八十六天对selenium模块进行较详细的了解

    孤荷凌寒自学python第八十六天对selenium模块进行较详细的了解 (今天由于文中所阐述的原因没有进行屏幕录屏,见谅) 为了能够使用selenium模块进行真正的操作,今天主要大范围搜索资料进行 ...