闭包

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

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

  • 必须有一个内嵌函数。
  • 内嵌函数必须引用外部函数中的变量。
  • 外部函数返回值必须是内嵌函数的引用。
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. vue-cli 引用elementUI打包后文件过大

    解决方案:使用externals引用第三方资源,防止element资源被打包到自己项目中,(总共修改3个页面index.html.webpack.base.conf.js.main.js) 1.修改i ...

  2. ES6-Symbol的用法 ,symbol在对象中的应用,改变值

    ES6-Symbol的用法,,symbol在对象中的应用,改变值 let a = new String; let b = new Number; let c = new Boolean; let d ...

  3. CSS学习笔记-动画模块

    动画模块:    1.过渡和动画之间的异同        1.1不同点            (1)过渡必须人为触发才能执行            (2)动画不需要人为触发就可以执行        1 ...

  4. JavaScript动态加载script方式引用百度地图API,Uncaught ReferenceError: BMap is not defined

    百度地图官网文档介绍使用JSSDK时,仅提供了2种引入方式: script引入 异步加载 实际工作场景中仅某一两个页面或者只是单纯有功能需要用到百度地图,所以没有必要在 index.html 中全局引 ...

  5. 一文解读MPA/SPA(转)

    应用模式 模式示意图 多页面应用 每一次页面跳转的时候,后台服务器都会返回一个新的html文档,这种类型的网站也就是多页网站,也叫多页应用. 页面跳转: 返回HTML优点: 首屏时间快,SEO效果好缺 ...

  6. 使用User-Agent防止HttpClient发送http请求时403 Forbidden和安全拦截

    问题的抛出 今天有客户反映,批付交易完成后,在我方服务器以“服务器点对点通信”的方式通知请求对方服务器时,对方拦截了请求.并贴了一张截图. 从截图可以看出来,对方拦截了我们的user-agent(Ap ...

  7. TP5 ajax请求报500错误

    场景:几个站点从阿里云迁移到腾讯云,然后 TP5项目 ajax请求报500错误 数据返回成功,但是http状态码是500,不走success,一直走error 如下图: 原因分析: 服务器centos ...

  8. html5+css3画太极并添加动画效果

    可兼容移动端视图 效果图如下:太极图是可以旋转的 具体实现如下: <!DOCTYPE html> <html lang="zh"> <head> ...

  9. python配置yaml

    我们在做自动化的过程中无论是接口自动化还是UI自动化都会存在很多数据,我们对于自动化中如何存放这些数据也是很重要一点,如果写在代码里的话,每次更换数据就有点繁琐,我们可以通过一个文件存放这些数据,然后 ...

  10. Linux下搭建及配置禅道服务器详细过程-包含软件资源-Dotest-董浩

    Linux环境下搭建禅道管理工具 1:百度云盘下载: 禅道--链接:https://pan.baidu.com/s/1Stu7nOZVIPO5TnpJWjWtiQ 提取码:dnik CentOs操作系 ...