闭包

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

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

  • 必须有一个内嵌函数。
  • 内嵌函数必须引用外部函数中的变量。
  • 外部函数返回值必须是内嵌函数的引用。
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. Python虚拟环境导出包安装到另一台电脑的方法

    环境:Windows 版 Python Python 3.6.8可以用自带的命令建立虚拟环境,不用安装另外的如 virtualenv.virtualenvwrapper等工具. python -m v ...

  2. Linux:Apache服务器的搭建

    下载安装并启动apache服务 安装apache服务 yum install -y httpd 启动apache服务 systemctl start httpd.service apache服务器的目 ...

  3. Linux下基于shell脚本实现学生信息管理系统

    #该管理系统是参考两位博主(时间有点远了,我忘了,请博主看到后联系我)后自行修改添加的.登录过程还有很多不完善,我就抛砖引玉啦. 废话不多,直接上码! #!/bin/bash# 学生管理系统# @ve ...

  4. 原型链 | 显示原型、隐式原型 | 构造关系constructor | instanceof

    1.原型关系 prototype 显式原型:prototype 隐式原型:__proto__ 1. 每个函数function都有一个prototype,即显式原型(属性)2. 每个实例对象都有一个__ ...

  5. AtCoder Grand Contest 040

    Preface 今年准备省选啥都不说了,省选题基本上都做过一遍了,开始尝试板刷AGC 这场做完就从AGC001开始吧,感觉以我的速度和来机房的频率一个礼拜做一场都谢天谢地了 A - >< ...

  6. plsql基础练习题

    1.键盘输入一个年份,判断是否是闰年; (能被4整除而不能被100整除或者能被100和400同时整除,满足其一即可); 方法1 declare v_year number(4):=&请输入一个 ...

  7. nodejs环境下的socket通信

    结构: socket是应用层和传输层的桥梁.(传输层之上的协议所涉及的数据都是在本机处理的,并没进入网络中) 涉及数据: socket所涉及的数据是报文,是明文. 作用: 建立长久链接,供网络上的两个 ...

  8. error while loading shared libraries

    https://stackoverflow.com/questions/480764/linux-error-while-loading-shared-libraries-cannot-open-sh ...

  9. Java设计模式:Flyweight(享元)模式

    概念定义 享元(Flyweight)模式运用共享技术高效地支持大量细粒度对象的复用. 当系统中存在大量相似或相同的对象时,有可能会造成内存溢出等问题.享元模式尝试重用现有的同类对象,如果未找到匹配的对 ...

  10. 【linux】切换到root用户,并重置root用户密码

    1.切换当前用户 到 root用户 sudo -i 2.重置root用户密码 sudo passwd root