闭包

定义:内部函数对外部函数变量的引用,则将该函数与用到的变量称为闭包。

闭包必须满足以下三个条件:

  • 必须有一个内嵌函数。
  • 内嵌函数必须引用外部函数中的变量。
  • 外部函数返回值必须是内嵌函数的引用。
def func(num):
def func_in(m):
print num, m # 结果:10 3
new_num = num ** m
return new_num
return func_in if __name__ == '__main__':
ret = func(10)
res = ret(3)
print res # 结果:1000

说明:func_in指向func_in()函数,return func_in 将函数的引用返回,用ret接收了这个返回值,ret就指向了func_in所指向的函数体,即func_in()函数。最后调用执行ret所指的函数。这就是闭包的整个过程,func_in()函数以及该函数内用到的变量num就称为闭包。简单说就是如果一个内嵌函数访问了外部嵌套函数作用域内的变量,则这个内嵌函数和用到的变量就称为闭包。将内嵌函数的语句和这些语句的执行环境打包在一起后,得到的函数对象称为闭包。

装饰器

装饰器是闭包的一种使用场景,在定义装饰器时需要传入一个函数对象,在此函数执行之前或者之后都可以追加其它的操作。

装饰器本质上是一个返回函数的高阶函数,装饰器接收要增强的函数,然后在装饰器内部进行功能增强。让函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象。

使用场景:性能测试,插入日志,事务管理,权限校验...。就好比是一个切面,也就是我们之后学习中会提到的叫面向切面编程(aop)。

开放封闭原则:开放表现在不改动源码(破坏原本业务逻辑)的同时扩展新的功能。封闭表现在不允许随意去修改源代码。

def outer(fun):
def inner():
print "功能开始前记录日志"
fun()
print "功能结束后记录日志"
return inner def work():
print "hello world" work = outer(work)
work() # 结果打印三行,分别是:功能开始前记录日志 hello world 功能结束后记录日志

无参无返回值的装饰器,示例如下

def outer(fun):
def inner():
print "功能开始前记录日志"
fun()
print "功能结束后记录日志"
return inner @outer # @outer 的效果等同于 work = outer(work),此时work变量指向inner()函数,work()即调用inner()函数。
def work():
print "hello world" work() # 结果打印三行,分别是:功能开始前记录日志 hello world 功能结束后记录日志

无参有返回值的装饰器,示例如下

def make_one(fun):
def wrapper():
return "===" + fun() + "==="
return wrapper def make_two(fun):
def inner():
return "---" + fun() + "---"
return inner @make_one
@make_two
def work():
return "hello world" print work() # 结果:===---hello world---===
# 首先执行装饰器make_two,即work = make_two(work)。这时work变量指向inner()函数。inner()函数里面的fun()函数指向work()函数。
# 然后执行装饰器make_one,即work = make_one(work)。这时work变量指向wrapper()函数。wrapper()函数里面的fun()函数指向inner()函数。
# 最后执行work()的时候,先执行wrapper()函数,运行里面的fun()函数时候,即执行inner()函数,返回"---hello world---"。

有参有返回值的装饰器,示例如下

def make_one(fun):
def wrapper(name, age, sex):
fun(name, age, sex)
return "有参有返回值的装饰器,返回值啦"
return wrapper @make_one
def work(name, age, sex):
print "我叫:%s,年龄:%s,性别:%s" % (name,age,sex) print work("小刘",27,"男")
# 运行结果:
# 我叫:小刘,年龄:27,性别:男
# 有参有返回值的装饰器,返回值啦

通用装饰器,示例如下

# 无参数的装饰器
def make_one(func):
def wrapper(*args, **kwargs):
print args, kwargs
return func(*args, **kwargs)
return wrapper @make_one
def work():
print "Hello world" work()
# 输出结果:
# () {}
# Hello world
# 有参数的装饰器
def make_one(func):
def wrapper(*args, **kwargs):
print args, kwargs
return func(*args, **kwargs)
return wrapper @make_one
def work(name, age, sex):
print "我叫:%s,年龄:%s,性别:%s" % (name,age,sex) work("小刘", 27, "男")
# 输出结果:
# ('\xe5\xb0\x8f\xe5\x88\x98', 27, '\xe7\x94\xb7') {}
# 我叫:小刘,年龄:27,性别:男
# 有参数有返回值的装饰器
def make_one(func):
def wrapper(*args, **kwargs):
print args, kwargs
res = func(*args, **kwargs)
return res # work函数的返回值
return wrapper @make_one
def work(name, age, sex):
return "我叫:%s,年龄:%s,性别:%s" % (name,age,sex) res = work("小刘", 27, "男") # 这里work变量指向wrapper()函数,func变量指向被装饰的真正的work()函数。
print res
# 运行结果:
# ('\xe5\xb0\x8f\xe5\x88\x98', 27, '\xe7\x94\xb7') {}
# 我叫:小刘,年龄:27,性别:男

附加:当我们定义一个函数的时候,其实也可以理解为我们定义了一个函数变量,我们可以将其作为值赋值给一个变量,或者当作一个方法参数传递。Python中函数也是一个对象,并且可以被直接赋值给变量,就可以通过该变量调用该函数。

参考:https://www.toutiao.com/i6700739314055119367/?tt_from=weixin&utm_campaign=client_share&wxshare_count=1&timestamp=1570675241&app=news_article&utm_source=weixin&utm_medium=toutiao_android&req_id=201910101040410100140470382201FC1A&group_id=6700739314055119367

Python—闭包和装饰器的更多相关文章

  1. python 闭包和装饰器

    python 闭包和装饰器 一.闭包闭包:外部函数FunOut()里面包含一个内部函数FunIn(),并且外部函数返回内部函数的对象FunIn,内部函数存在对外部函数的变量的引用.那么这个内部函数Fu ...

  2. python闭包与装饰器

    转自小马哥: 闭包和装饰器充分体现了Python语法糖的优雅感觉. 在本文中,我们的实验要完成两个工作,一个是加法,一个是累计调用加法的次数,最普通的Python程序可以这么写: def valida ...

  3. 高逼格利器之Python闭包与装饰器

    生活在魔都的小明,终于攒够了首付,在魔都郊区买了一套房子:有一天,小明踩了狗屎,中了一注彩票,得到了20w,小明很是欢喜,于是想干脆用这20万来装修房子吧(decoration): 整个装修过程,小明 ...

  4. Python 简明教程 --- 22,Python 闭包与装饰器

    微信公众号:码农充电站pro 个人主页:https://codeshellme.github.io 当你选择了一种语言,意味着你还选择了一组技术.一个社区. 目录 本节我们来介绍闭包与装饰器. 闭包与 ...

  5. Python闭包及装饰器

    Python闭包 先看一个例子: def outer(x): def inner(y): return x+y return innder add = outer(8) print add(6) 我们 ...

  6. python闭包以及装饰器

    通俗的定义:如果在一个内部函数里,对在外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认为是闭包(closure).它只不过是个“内层”的函数,由一个名字(变量)来指代,而这个名字(变 ...

  7. python闭包和装饰器

    本文目录: 1. 闭包的解析和用法 2. 函数式装饰器 3. 类装饰器 一.闭包 闭包是一种函数,从形式上来说是函数内部定义(嵌套)函数,实现函数的扩展.在开发过程中,考虑到兼容性和耦合度问题,如果想 ...

  8. python闭包和装饰器(转)

    一.python闭包 1.内嵌函数 >>> def func1(): ... print ('func1 running...') ... def func2(): ... prin ...

  9. 详解Python闭包,装饰器及类装饰器

    在项目开发中,总会遇到在原代码的基础上添加额外的功能模块,原有的代码也许是很久以前所写,为了添加新功能的代码块,您一般还得重新熟悉源代码,稍微搞清楚一点它的逻辑,这无疑是一件特别头疼的事情.今天我们介 ...

随机推荐

  1. hadoop免登录

    参考:http://wenku.baidu.com/link?url=n4PT7AhGnV7N8KevSEAMcCVGEaYqTuKmNodCQsUnR7qtAnWM0WDs8pFYLOpCUu9R9 ...

  2. Java中的“scanf()、cin()、input()"

    最近在写一个Java程序时遇到一个问题,就是如何在Java里面输入数值,又叫做获取键盘输入值. 因为c语言里面有scanf(),C++里面有cin(),python里面有input().Java里面有 ...

  3. Windows 7下Node.js Web开发环境搭建笔记

    Node.js是什么? 我们看看百科里怎么说的?JavaScript是一种运行在浏览器的脚本,它简单,轻巧,易于编辑,这种脚本通常用于浏览器的前端编程,但是一位开发者Ryan有一天发现这种前端式的脚本 ...

  4. MySQL问题记录——2003-Can't connect to MySQL server on 'localhost'(10038)

    MySQL问题记录——2003-Can't connect to MySQL server on 'localhost'(10038) 摘要:本文主要记录了连接到MySQL数据库时出现的问题以及解决办 ...

  5. 如何将RAC数据库的 RMAN Disk 备份 Restore 到另一个节点上的单个实例 (Doc ID 415579.1)

    HowTo Restore RMAN Disk backups of RAC Database to Single Instance On Another Node (Doc ID 415579.1) ...

  6. 学习postman教程

    postman可以做什么 1.可以做单接口的测试 2.可以调试接口 3.对接口设置变量后,可以做多接口的测试,并输出报告 如何用postman调试接口 1.填写api地址 2.选择请求方式 3.输入a ...

  7. npm --save-dev 和 --save 的区别

    转载 >>> 1. npm install 在安装 npm 包时,有两种命令参数可以把它们的信息写入 package.json 文件, 一个是npm install--save    ...

  8. 第05组 Beta冲刺(1/4)

    第05组 Beta冲刺(1/4) 队名:天码行空 组长博客连接 作业博客连接 团队燃尽图(共享): GitHub当日代码/文档签入记录展示(共享): 组员情况: 组员1:卢欢(组长) 过去两天完成了哪 ...

  9. 2019.10.02模拟赛T3

    题目大意: 设$S(n,m)$为第二类斯特林数,$F_i$表示斐波那契数列第$i$项. 给定$n,R,K$,求$\sum\limits_{i=1}^{n}(\sum\limits_{m=1}^{R}F ...

  10. spring cloud 2.x版本 Spring Cloud Stream消息驱动组件基础教程(kafaka篇)

    本文采用Spring cloud本文为2.1.8RELEASE,version=Greenwich.SR3 本文基于前两篇文章eureka-server.eureka-client.eureka-ri ...