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. Java:多线程中的volatile

    一.为什么使用volatile 首先,通过一段简单的代码来理解为什么要使用volatile: public class RunThread extends Thread{ private boolea ...

  2. mysql 常用语句使用

    1.查询语句 SELECT * FROM table 2.更改语句 UPDATE table SET name = '123456' WHERE id = 100 3.插入语句 INSERT INTO ...

  3. PLSQL-包函数存储过程

    包: 包是PLSQL中多个单元的逻辑组合,他将过程组合在一个包内容,以供用户调用,使用后,不需要程序员频繁的修改程序,可以保持程序的逻辑完整性,对包中的过程重新定义或者编译,以便修改部分功能,从而更好 ...

  4. hackhttp模板的介绍

    hackhttp模板:造福人类 发起get/post/ 发起http原始数据包 漏洞利用:更为快捷放放不安 #hackhttp使用方法hh=hackhttp.hackhttp() code,head, ...

  5. Qt-QSplashScreen-程序启动动画

    多数大型应用程序启动时可会在程序完全启动前显示一个启动画面,在程序完全启动后消失,程序启动画面可以显示相关产品的一些信息,使用户在等待程序启动时同时了解产品的相关功能,这也是一种宣传方式. 首先运行界 ...

  6. 用python读取配置文件config.ini

    还在学习中...写的有点凌乱 感觉还是应该先学会读取配置文件才行,把一些经常需要修改的但是又经常需要用到的参数放到配置文件中方便使用(我是这么觉得的) 首先是config.ini的存放位置,我们把它放 ...

  7. POJ 3256 (简单的DFS)

    //题意是 K N, M; //有K个牛 N个牧场,M条路 ,有向的  //把K个牛放到任意的n个不同牧场中,问所有牛都可以到达的牧场数的总和  //这是一道简单的DFS题 //k 100 //n 1 ...

  8. Spring Cloud(八):配置中心(服务化与高可用)【Finchley 版】

    Spring Cloud(八):配置中心(服务化与高可用)[Finchley 版]  发表于 2018-04-19 |  更新于 2018-04-26 |  本文接之前的<Spring Clou ...

  9. JavaScript 的一些基础知识

    JavaScript基本语法 调试 打开 Chrome 开发工具 Win F12 Mac Command + Option + I 输入代码.测试执行 var str = 'evenyao' cons ...

  10. docker学习2

    今天继续学习docker! 搜索镜像 docker search centos 下载镜像 docker pull name(镜像名字) 查看镜像docker images 字段含义分析: TAG:仓库 ...