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. 上海Uber优步司机奖励政策(1月4日~1月10日)

    滴快车单单2.5倍,注册地址:http://www.udache.com/ 如何注册Uber司机(全国版最新最详细注册流程)/月入2万/不用抢单:http://www.cnblogs.com/mfry ...

  2. CSS3中的动画

    CSS3中的动画包括两种: Transition(过渡) Animation(动画) 这两种方法都可以让元素动起来,功能类似,但是稍有区别: Transition只定义某一个元素的开始状态和结束状态 ...

  3. 14、Java并发编程:CountDownLatch、CyclicBarrier和Semaphore

    Java并发编程:CountDownLatch.CyclicBarrier和Semaphore 在java 1.5中,提供了一些非常有用的辅助类来帮助我们进行并发编程,比如CountDownLatch ...

  4. unity share current game screen

    using UnityEngine; using System.Collections; using UnityEngine.UI; using System.IO; public class Tak ...

  5. 腾讯WeTest开启“测试扶持计划”赠送重磅福利(含MTSC/TiD门票)

    WeTest导语 伴随着互联网行业的发展,与各行各业的连接更加紧密,竞争也变得越发激烈,用户对于产品的体验开始变得更加“挑剔”.然而目前互联网产品却始终受到各类质量问题的困扰.以兼容问题为例,应用平台 ...

  6. Selenium 入门到精通系列:一

    Selenium 入门到精通系列 PS:控制浏览器窗口大小.前进.后退.刷新 例子 #!/usr/bin/env python # -*- coding: utf-8 -*- # @Date : 20 ...

  7. Java开发工程师(Web方向) - 01.Java Web开发入门 - 第1章.Web应用开发概述

    第1章--Web应用开发概述 Web应用开发概述 浏览器-服务器架构(BS-architecture) browser/ App    ---- request ---->    server ...

  8. 剑指offer-二叉树搜索树与双向链表25

    题目描述 输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表.要求不能创建任何新的结点,只能调整树中结点指针的指向. class Solution: def Convert(self, pRo ...

  9. Period :KMP

    I - Period Problem Description For each prefix of a given string S with N characters (each character ...

  10. java超强分页标签演示

    最近在做一个项目,用到了一个分页,于是动手写了个分页标签,先将代码贴出来,供大家交流,写的不好,请见谅!. 以下是java标签类,继承自SimpleTagSupport package com.lyn ...