一、嵌套函数

函数的内部又再定义另一个函数,这个函数就叫嵌套函数,里面含函数就叫内部函数。

示例:

二、返回函数

函数可以接收函数对象作为参数,同理函数也能返回一个函数对象作为返回值。

示例:

返回函数可以用来延迟函数的执行。

三、命名空间与变量作用域

变量作用域指的是变量的存活的范围。命名空间指的是属于一个对象的所有属性(对象)的集合。

示例:

A的命名空间是A函数对象里面的所有对象的集合,包括变量a、函数B、变量b;B的命名空间就是属于函数B的所有对象的集合,包括变量b;

a的变量作用域就是A的命名空间,b的变量作用域就是B的命名空间。

可以看出,命名空间就是对象的下一层的全部空间(比如A的命名空间就是A的下一层全部空间),变量作用域就是变量所处层的上一层对象的命名空间(比如a的作用域就是A的命名空间)。

四、闭包

一个嵌套函数的内部函数使用了内部函数以外的变量,这个嵌套函数就叫闭包。

示例:

闭包给我们一个重要的提示是,一个变量只要在作用域里面,它可以在内部函数中发挥作用。

五、装饰器

装饰器本质是一个函数,是一个返回函数的高阶函数,也是一个闭包。装饰器能解决的问题是:在一个原函数功能不变的情况下,为这个原函数增加一些新的功能。

比如说有一个打印我的名字的函数myname:

现在想在不改变myname的代码的情况下打印i come from foshan,这样就用到装饰器了:

事实上,我们不需要内部函数也可以为传入函数添加功能的:

这样做的话如果我们传入的func带有参数就用不了了,我们还可以这样:

这样的话虽然能传入带有参数的func,但新增加的功能与func的调用分开,这样也不妥。

所以我们的装饰器应该是集接收函数、返回函数、闭包为一身的函数,而且允许传入的函数带参数:

(*args,**kw)参数组表示可以传入任意的参数。

六、装饰器操作符@

定义了装饰器之后,我们就可以拿装饰器去应用在任意我们想增加相应功能的函数了:

为了使装饰器使用起来时更方便,主要为了代码看上去逻辑更清晰明了,于是便定义了@能直接调用参数,具体用法如下:

在函数定义前声明了@wherefrom这个语句后,在函数调用时就会执行一下逻辑:

myname = wherefrom(myname)

七、高级装饰器

我们在定义装饰器时,原函数是不能更改的。如果我们新增加的功能中还要网里面传入新参数,这种装饰器该怎么定义?思路很简单,本来两层的嵌套函数我们写够三层就可以了,第一层用于传入新参数,第二层用于传入函数:

(注意:这里的aplace和func传入的顺序不能改变,请想一下为什么。)

测试一下效果:

这里执行逻辑是:

myname = wherefrom("guangzhou")(myname)

八、装饰后的函数名字问题

无论是普通装饰器还是高级装饰器,在用了@这个操作符装饰函数后,在调用被装饰函数的时候,其函数名指向的具体的函数对象已经不是原来的函数对象了,就像上面的:myname = wherefrom(myname) 和 myname = wherefrom("guangzhou")(myname) 一样,函数名都指向了inner这个函数对象:

但有时候我们后续程序里可能会用调用到原函数的名字,如上面的myname.__name__这样,为了避免发生混淆,我们应该在装饰器定义时考虑这个事情:

我们在inner函数的代码毫无改动的情况下,把inner函数的名字指向了func函数的名字,所以我们能不能把这种新增功能定义成一个装饰器,用来提供给其他的装饰器定义时使用呢?python内部已经帮我们实现了这种功能:

functools中的wraps装饰器就是专门做这个的。有兴趣的可以看看wraps的源代码,看这个装饰器怎么实现的。

具体参考:

1、《核心编程第二版》第11章;

2、廖雪峰 - 函数式编程

——————本篇完!

13、python中的函数(闭包与装饰器)的更多相关文章

  1. Python之面向对象:闭包和装饰器

    一.闭包 1. 如果一个函数定义在另一个函数的作用域内,并且引用了外层函数的变量,则该函数称为闭包. def outter(): name='python' def inner(): print na ...

  2. python基础之函数当中的装饰器

    在实际工作当中存在一个开放封闭原则 1.对扩展是开放的 为什么要对扩展开放呢? 我们说,任何一个程序,不可能在设计之初就已经想好了所有的功能并且未来不做任何更新和修改.所以我们必须允许代码扩展.添加新 ...

  3. python中对 函数 闭包 的理解

    最近学到 函数 闭包的时候,似懂非懂.迷迷糊糊的样子,很是头疼,今天就特意查了下关于闭包的知识,现将我自己的理解分享如下! 一.python 闭包定义 首先,关于闭包,百度百科是这样解释的: 闭包是指 ...

  4. python学习总结---函数使用 and 装饰器

    # 函数使用 ### 零碎知识 - 灵活的if-else ```python a = 3 if False else 5 print(a) ''' if False: a = 3 else: a = ...

  5. Python之命名空间、闭包、装饰器

    一.命名空间 1. 命名空间 命名空间是一个字典,key是变量名(包括函数.模块.变量等),value是变量的值. 2. 命名空间的种类和查找顺序 - 局部命名空间:当前函数 - 全局命名空间:当前模 ...

  6. Python 变量作用域,闭包和装饰器

    from dis import dis b = 6 def f1(a): print(a)print(b) b = 9 f1(3) print(dis(f1)) # dis模块可以查看python函数 ...

  7. python基础编程: 函数示例、装饰器、模块、内置函数

    目录: 函数示例 装饰器 模块 内置函数 一.函数示例: 1.为什么使用函数之模块化程序设计: 不使用模块程序设计的缺点: 1.体系结构不清晰,可主读性差: 2.可扩展性差: 3.程序冗长: 2.定义 ...

  8. day20 函数闭包与装饰器

    装饰器:本质就是函数,功能是为其他函数添加新功能 原则: 1.不修改被装饰函数的源代码(开放封闭原则) 2.为被装饰函数添加新功能后,不修改被修饰函数的调用方式 装饰器的知识储备: 装饰器=高阶函数+ ...

  9. Python中利用函数装饰器实现备忘功能

    Python中利用函数装饰器实现备忘功能 这篇文章主要介绍了Python中利用函数装饰器实现备忘功能,同时还降到了利用装饰器来检查函数的递归.确保参数传递的正确,需要的朋友可以参考下   " ...

随机推荐

  1. cf1059D. Nature Reserve(三分)

    题意 题目链接 Sol 欲哭无泪啊qwq....昨晚一定是智息了qwq 说一个和标算不一样做法吧.. 显然\(x\)轴是可以三分的,半径是可以二分的. 恭喜你获得了一个TLE的做法.. 然后第二维的二 ...

  2. canvas制作倒计时效果

  3. react-native与原生界面相互跳转

    一.添加MyIntentModule类,并继承ReactContextBaseJavaModule实现其方法和构造函数.在该类中添加方法,注意:方法头要加@ReactMethod public cla ...

  4. 关于 Android Studio 如何连接手机调试

    第一步:设置-> 打开开发者选项,以及USB调试模式 第二步:关于手机->版本号,点击版本号会弹出提示:已经处于开发者模式,无需操作 第三步:设置->在搜索框中输入HDB,此时会弹出 ...

  5. April 13 2017 Week 15 Thursday

    Happiness takes no account of time. 幸福不觉光阴过. Do you know the theory of relativity? If you know about ...

  6. 【css基础修炼之路】— 谈谈元素的垂直水平居中

    作为一个初级的前端工程师,在开发的过程中遇到了许多问题,其中使元素垂直居中这个问题难住了我,可能在大家看来这是一个非常小的问题,但是却困扰了我很长时间,于是决定做一个总结!!! 废话不多说,直接上代码 ...

  7. springboot实现邮件发送

    1.创建springboot项目. 2.创建好的项目如图: 在static目录下新建index.html. 3.点击启动项目 在浏览器的地址栏中访问:http://localhost:8080/ 访问 ...

  8. a=a+(a++);b=b+(++b);计算顺序,反汇编

    a=a+(a++); 013913BC mov eax,dword ptr [a] 013913BF add eax,dword ptr [a] 013913C2 mov dword ptr [a], ...

  9. Win7多用户同时登陆

    软件提供下载: http://pan.baidu.com/s/1o6FQv70

  10. apache配置局域网访问

    1.配置vhost.conf NameVirtualHost 192.168.2.74:80 <VirtualHost 192.168.2.74:80> DocumentRoot /var ...