一. 一切皆对象

函数式编程并没有标准定义,如果代码非常繁琐则考虑使用。

学习闭包的概念,不是python独有的。

其他大多数语言中的函数只是一段可执行的代码,并不是对象。

python中的函数是对象,一切皆对象。可以把函数赋值给变量:

a = 1

a = '2'

a = def

甚至可以把函数当作另外一个函数的参数传递或者当成返回值返回,而C#中要封装成委托。

二.什么是闭包:闭包=函数+函数定义时的环境变量

我们尝试从概念上去理解一下闭包。

在一些语言中,在函数中可以(嵌套)定义另一个函数时,如果内部的函数引用了外部的函数的变量,则可能产生闭包。闭包可以用来在一个函数与一组“私有”变量之间创建关联关系。在给定函数被多次调用的过程中,这些私有变量能够保持其持久性。—— 维基百科

用比较容易懂的人话说,就是当某个函数被当成对象返回时,夹带了外部变量,就形成了一个闭包。

1.

code1

def curve_pre():

    def curve():

        pass

curve()    #找不到,因为curve()的作用域仅限于curve_pre()的内部

code2

def curve_pre():

    def curve():
print('This is a function')         pass     return curve    #函数可以作为结果返回 f = curve_pre()    #函数可以赋值 f()#执行函数,不会报错
#This is a function

code3

def curve_pre():

    a = 25         #局部变量在curve的外部

    def curve(x):  #接受抛物线的x值

        return a * x * x

    return curve    #返回一个函数

f = curve_pre()

print(f(2))         #调用curve()函数

#

code4

a = 10

def f1(x):

    return a * x * x

print(f1(2))

#40         #局部变量找不到会去上一级找

code 3 和 code4 对比

def curve_pre():

    a = 25 #局部变量在curve的外部

    def curve(x): #接受抛物线的x值

        return a * x * x

    return curve #返回一个函数

a = 10     #定义a = 10

f = curve_pre()

print(f(2)) #调用curve()函数

#100        #仍然是a = 25的取值,取得是定义时的环境变量,这就是闭包

函数及其外部环境变量所构成的整体叫做闭包

2.环境变量要在函数外部,但不能是全局变量:

a = 25    #a定义为了全局变量

def curve_pre():

    def curve(x): #接受抛物线的x值

        return a * x * x

    return curve #返回一个函数

a = 10

f = curve_pre()

print(f(2)) #调用curve()函数

#40    #a的值被改变了,因为没有形成闭包

查看:

def curve_pre():

    a = 25 #局部变量在curve的外部

    def curve(x): #接受抛物线的x值

        return a * x * x

    return curve #返回一个函数

a = 10

f = curve_pre()

print(f.__closure__)

print(f.__closure__[0].cell_contents)#取出闭包的环境变量

print(f(2)) #调用curve()函数

#(<cell at 0x0031AAF0: int object at 0x0FF93A80>,)

#25    #这里是a的值

#

三.一个事例看看闭包

1.闭包的意义:闭包保存的是一个环境,把函数现场保存起来了。

闭包 = 函数 + 函数定义时的环境变量(不能是全局变量)

2. 闭包的经典误区      从外层向内层分析

def f1():

    a = 10

    def f2():

        a = 20    #对a重新赋值局部变量,不会影响到外部

        print(a)

    print(a)    #

    f2()    #20 #f1内部调用f2

    print(a)    #

f1()

#

#

#

验证其是不是一个闭包:

1

def f1():

    a = 10

    def f2():

        a = 20

        # print(a)

    #print(a)

    f2()

    #print(a)

f = f1()    #f1是None类型

print(f.__closure__)    #报错

(2)加上返回值,仍然不是闭包函数:

def f1():

    a = 10

    def f2():

        a = 20    #a被认为是一个局部变量了,就不认为是个环境变量了

        return a

    return f2

f = f1()

print(f.__closure__)    #没有__closure__属性

#None

(3)去掉f2()中的赋值后是闭包函数:

def f1():

    a = 10

    def f2():

        #a = 20    #删除a = 20,此时a被python认为是一个局部变量

        return a

    return f2

f = f1()

print(f.__closure__)

#(<cell at 0x02F5AAF0: int object at 0x0FF93990>,)

原因:环境变量不能当作一个变量去赋值,而是一定要去引用外部。

四.出个题,用闭包解决!

闭包不是必不可少的东西,只是可以使你的代码架构更加合理。

题目:

旅行者,x = 0 为起点,没走一步加1,计算出旅行者当前所处的位置。

关键点:每次调用时需要调用上次的结果

1.先用非闭包解决一下

origin = 0

def go(step):

    global origin    #将origin变成全局变量

    new_pos = origin + step

    origin = new_pos  #等号左边的origin被python认识是局部变量

    return origin

print(go(2))

print(go(3))

print(go(6))

#

#

#

2.再用闭包解决一下

origin = 0

def factory(pos):    #工厂模式

    def go(step):

        nonlocal pos #强制声明不是局部变量

        new_pos = pos + step

        pos = new_pos

        return new_pos

    return go

tourist = factory(origin)

print(tourist(2))
print(origin) print(tourist(3)) print(tourist(6)) #
# # #

并没有改变全局变量origin的值

五.小谈函数式编程

  • python和Javascript都会使用闭包。
  • 闭包特点:在一个函数外部间接调用函数内部的变量,从模块级别间接调用局部变量。
  • 极易造成内存泄漏。

闭包的概念

Python3(九) 闭包的更多相关文章

  1. Python3基础 闭包 简单示例

    镇场诗: 诚听如来语,顿舍世间名与利.愿做地藏徒,广演是经阎浮提. 愿尽吾所学,成就一良心博客.愿诸后来人,重现智慧清净体.-------------------------------------- ...

  2. Swift5 语言指南(九) 闭包

    闭包是自包含的功能块,可以在代码中传递和使用.Swift中的闭包类似于C和Objective-C中的块以及其他编程语言中的lambdas. 闭包可以从定义它们的上下文中捕获和存储对任何常量和变量的引用 ...

  3. python3 之 闭包实例解析

    一.实例1: def make_power(y): def fn(x): return x**y return fn pow3 = make_power(3) pow2 = make_power(2) ...

  4. python学习之路 五:函数式编程

    本节重点 掌握函数的作用.语法 掌握作用域.全局变量与局部变量知识 掌握函数名称空间.闭包 一.函数编程基础知识 1.基本定义 函数是指将一组语句的集合通过一个名字(函数名)封装起来,要想执行这个函数 ...

  5. Web前端经典面试试题(一)

    本篇收录了一些面试中经常会遇到的经典面试题,并且都给出了我在网上收集的答案.眼看新的一年马上就要开始了,相信很多的前端开发者会有一些跳槽的悸动,通过对本篇知识的整理以及经验的总结,希望能帮到更多的前端 ...

  6. python3.4学习笔记(十九) 同一台机器同时安装 python2.7 和 python3.4的解决方法

    python3.4学习笔记(十九) 同一台机器同时安装 python2.7 和 python3.4的解决方法 同一台机器同时安装 python2.7 和 python3.4不会冲突.安装在不同目录,然 ...

  7. python3.4学习笔记(九) Python GUI桌面应用开发工具选择

    python3.4学习笔记(九) Python GUI桌面应用开发工具选择 Python GUI开发工具选择 - WEB开发者http://www.admin10000.com/document/96 ...

  8. python3 练习题100例 (九)

    题目九:题目:古典问题:有一对兔子,从出生后第3个月起每个月都生一对兔子,小兔子长到第三个月后每个月又生一对兔子,假如兔子都不死,问每个月的兔子总数为多少?程序分析:兔子的规律为数列1,1,2,3,5 ...

  9. python3 练习题100例 (十九)

    #!/usr/bin/env python3 # -*- coding: utf-8 -*- """练习十九:计算1-2+3...+99中除了88以外所有数的和" ...

随机推荐

  1. 带限制的广搜 codeforces

    You are playing some computer game. One of its levels puts you in a maze consisting of n lines, each ...

  2. Redis(二):redis命令构建及关键属性解析

    上一篇文章,我们从框架层面,主要介绍了redis的启动过程,以及主要的命令处理流程逻辑.这些更多的都是些差不多的道理,而要细了解redis,则需要更细节的东西. 今天我们稍微内围的角度,来看看几个命令 ...

  3. Java入门 - 语言基础 - 21.Scanner类

    原文地址:http://www.work100.net/training/java-scanner.html 更多教程:光束云 - 免费课程 Scanner类 序号 文内章节 视频 1 概述 2 使用 ...

  4. maven版本对应的jdk

    今天配置项目环境发现jdk1.6与maven-3.39不能匹配 查询jdk与maven的版本对应关系 关系网址:http://maven.apache.org/docs/history.html Ma ...

  5. 异步查询转同步加redis业务实现的BUG分享

    在最近的性能测试中,某一个查询接口指标不通过,开发做了N次优化,最终的优化方案如下:异步查询然后转同步,再加上redis缓存.此为背景. 在测试过程中发现一个BUG:同样的请求在第一次查询结果是OK的 ...

  6. Java服务端两个常见的并发错误

    理想情况来讲,开发在开始编写代码之前就应该讲并发情况考虑进去,但是大多数实际情况确是,开发压根不会考虑高并发情况下的业务问题.主要原因还是因为业务极难遇到高并发的情况. 下面列举两个比较常见的后端编码 ...

  7. 7、python基本数据类型之散列类型

    前言:python的基本数据类型可以分为三类:数值类型.序列类型.散列类型,本文主要介绍散列类型. 一.散列类型 内部元素无序,不能通过下标取值 1)字典(dict):用 {} 花括号表示,每一个元素 ...

  8. 大叔 EF 来分析 EntityFrameworks.Data.Core 2

    Extensions 1DbCommand拦截器扩展DbCommandInterceptorExtensions 2Class for IQuerable extensions methods Inc ...

  9. 异想家Eclipse的偏好配置

    1.汉化 http://www.eclipse.org/babel/downloads.php 找到Babel Language Pack Zips,下面选自己版本点进去,找到如下类似的中文包: Ba ...

  10. 本地python3环境下运行报错CV2的问题

    如上图,执行脚本后,报找不到指定的模块的错误 解决方法: 1.import cv2提示"Dll load failed:找不到指定的模块" 解决方法:那卸载掉opencv-cont ...