Python成长之路_装饰器
一、初入装饰器
- 1、首先呢我们有这么一段代码,这段代码假设是N个业务部门的函数
- def f1(aaa):
- print('我是F1业务')
- if aaa == 'f1':
- return 'ok'
- def f2(aaa):
- print('我是F2业务')
- if aaa == 'f2':
- return 'ok'
业务代码
这里的函数的意思就是当我们调用上面的函数的时候,传入值给aaa,当aaa的值等于f1或者f2就返回ok
2、公司有N个业务部门,1个基础平台部门,基础平台负责提供底层的功能,如:数据库操作、redis调用、监控API等功能。业务部门使用基础功能时,只需调用基础平台提供的功能即可。那么我们业务部门调用功能的时候只需要。
f1(值)
f2(值)
3、公司的运行正在有条不稳的进行,然而有一天呢,你的老大说,我发现了一个问题就是呢业务部调用基础平台的功能的时候没有验证这样不好所以呢老大就把工作交给了LOW们,要求增加验证功能并且业务部门在调用功能的方式不能变
Low A,这个A呢他是这么做的
他呢跟各个做基础功能的人协调,要求在自己的代码上加入验证模块,那么这样呢整个的基础平台就不需要更改,结果,Low A当天就被开除了……
Low B,这个B呢一看A都被开除了不行上面的方法不行哪就换一个
- def f1(aaa):
- #验证代码
- #验证代码
- print('我是F1业务')
- if aaa == 'f1':
- return 'ok'
- def f2(aaa):
- #验证代码
- #验证代码
- print('我是F2业务')
- if aaa == 'f2':
- return 'ok'
LowB
这家伙呢在每个功能前面加入了验证代码,然后过了两天Low B也被开除了
最后老大把工作交给了Low C
Low C呢总结了两个Low的教训他是这么干的
- #验证函数
- def verify():
- # 验证1
- # 验证2
- # 验证3
- pass
- def f1(aaa):
- verify():
- print('我是F1业务')
- if aaa == 'f1':
- return 'ok'
- def f2(aaa):
- verify():
- print('我是F2业务')
- if aaa == 'f2':
- return 'ok'
Lowc
- 他呢把验证功能的,写成了一个函数然后,每个业务模块来去调用
- 老大看见了LowC的实现方式,嘴角露出了一丝微笑,并且与LowC聊了个天
- 老大说:
写代码要遵循开发封闭原则,虽然在这个原则是用的面向对象开发,但是也适用于函数式编程,简单来说,它规定已经实现的功能代码不允许被修改,但可以被扩展,即:
- 封闭:已实现的功能代码块
- 开放:对扩展开发
如果将开放封闭原则应用在上述需求中,那么就不允许在函数 f1 、f2的内部进行修改代码,老板就给了Low C一个实现方案:
装饰器(单层装饰器)
- def out(main):
- def wra():
- # 验证1
- # 验证2
- # 验证3
- return xxx
- return wra
- @out
- def f1(aaa):
- print('我是F1业务')
- if aaa == 'f1':
- return 'ok'
- @out
- def f2(aaa):
- print('我是F2业务')
- if aaa == 'f2':
- return 'ok'
单层装饰器
二、好,故事(这个故事是盗取银角大王的)到此结束我们开始来看上面的代码
- (1)首先我们要知道当函数不加括号我们的函数是不被执行的,它会返回这个函数的内存地址
- def aaa():
- print("ok")
- print(aaa)
- <function aaa at 0x0000000000B6AC80>
函数内存地址
(2)然后我们来解释一下@out
@out就等于f1 = out(f1)
什么意思呢@out是python语法中的一个简写,他的用处就是针对装饰器来去做的,我们看下面的例子,将@out替换成
- f1 = out(f1)
- def out(main):
- def wra(aaa):
- print('我进来了')
- ccc = main(aaa)
- return ccc
- return wra
- def f1(aaa):
- print('我是F1业务')
- if aaa == 'f1':
- return 'ok'
- f1 = out(f1)
- s1 = 'f1'
- ff1 = f1(s1)
- print(ff1)
转换后
- (3)我们来解释下out函数,对于out来说他首先接收一个值,从上面可以看出他接收的值是f1,f2的内存地址,然后return返回函数wra,注意的是f1 = out(f1),没加括号所以都没执行,返回wra的时候也没加括号所以也没执行,那么到了这里这个函数就先暂停我们结合者调用来看下面的图:
- 第一步:请务必分开wra不等于f1而是main等于f1!!下面的f1就等于wra的意思是执行f1函数就等于执行wra函数
- 第二步:
- 最后呢我们通过python语法塘,将f1 = out(f1) 变成@out就变成了之前我们看到的代码
- def out(main):
- def wra(aaa):
- print('我进来了')
- ccc = main(aaa)
- return ccc
- return wra
- @out
- def f1(aaa):
- print('我是F1业务')
- if aaa == 'f1':
- return 'ok'
- s1 = 'f1'
- ff1 = f1(s1)
- print(ff1)
完整单层装饰器
友情提示:
- 在此请将之前的练习一下并且熟知其中的原理再进行下面的学习,大神除外
- (4)装饰器装上含有参数的函数,有同学可能测试了一下说,我把函数f1的参数变成了多个,装饰器就报错了,那是怎么回事呢,原因很简单,因为wra首先他只能接收一个参数,并且ccc = main(aaa)也只接收了一个参数
- def out(main):
- def wra(aaa):
- print('我进来了')
- ccc = main(aaa)
- return ccc
- return wra
单参数
既然知道了原因那么我们就给他改改数码宝贝超进化…:
- def out(main): #这里就不多解释了跟上面一样
- def wra(*aaa,**aa): #这里呢变成了啥是不是可以接收各种的参数了
- print('我进来了')
- ccc = main(*aaa,**aa) #这里呢也可以给f1各种参数
- return ccc
- return wra
- @out
- def f1(*af1):
- print('我是F1业务')
- print(af1[0]) #给大家测试用的
- print(af1[1])
- if af1[0] == 'f1':
- return 'ok'
- s1 = 'f1'
- ff1 = f1(s1,'我是参数2') #传入了两个值s1和我是参数2'
- print(ff1) #运行一下看看吧呵呵
装饰参数函数的装饰器
三、装饰器的终极进化(多层装饰器)逻辑比较绕
有一天,变态的老大又找到了Low C说你的装饰器练习的咋样了,Low C说:经过老大的调教,我已经练习的差不多了,这个时候老大阴阴的呵呵一笑,好,这样我呢又有个需求要你给我改改,我现在呢想在验证之后呢添加一个欢迎功能,这个功能呢,我们业务线的功能想要添加就添加先要不添加就不添加,要记住封闭原则哦0.0……….
第二天Low C找到了老大说,大哥啊您晚上还是来我家教教我吧,真心的不知道啊0.0,,,于是老大就去了Low C的家里经过一场风云(此次省略一万个字)0.0老大提供了另外的参考代码:
- def ds():
- print('ok我是欢迎信息1')
- def ss():
- print('ok我是欢迎信息2')
- def fill(*ill):
- def out(main):
- def wra(*waa,**wad):
- ill[0]()
- ill[1]()
- ccc = main(waa[0])
- return ccc
- return wra
- return out
- @fill(ds,ss)
- def f1(aaa):
- print('我是F1业务')
- if aaa == 'f1':
- return 'ok'
- c1 = f1('f1')
- print(c1)
老大的代码
- 哈哈看不懂了吧(大神除外)来吧我们分开来看就知道了
- 第一步:还是先解释一下装饰器fill请看图片,有一句话要牢记这个装饰器我就把它理解成调用,也就是把需要用的函数传入到装饰器做值,从而调用值来执行函数
![]()
- 第二步:
![]()
好到了这里大家应该明白了吧,不明白的留言,讲错了的请指教谢谢O(∩_∩)O~~,那么有的小伙伴要问了,这尼玛不对啊,为毛我把@fill(ds,ss)变成@fill(ss)就报错了呢?
我们分析一下问题,主要的原因呢就是def wra函数下面执行了 ill[1]报错了,因为这里*ill只有一个参数那么避免呢,看我的终极大招:
- def ds():
- print('ok我是欢迎信息1')
- def ss():
- print('ok我是欢迎信息2')
- def fill(*ill):
- def out(main):
- def wra(*waa,**wad):#这里加个判断不就完了么 O(∩_∩)O哈哈~
- if len(ill) != '':
- for i in range(0,len(ill)):
- ill[i]()
- ccc = main(waa[0])
- return ccc
- return wra
- return out
- @fill(ss,ds)
- def f1(aaa):
- print('我是F1业务')
- if aaa == 'f1':
- return 'ok'
- @fill() #你看这里没参数吧
- def f2(aaa):
- print('我是F2业务')
- if aaa == 'f2':
- return 'ok'
- c1 = f1('f1')
- print(c1)
- c2 = f2('f2')
- print(c2)
终极大招-多层装饰器
- #运行下试试吧
- 等等!最后老大和Low C成为了…此处省略一百万个字
作者:北京小远
出处:http://www.cnblogs.com/bj-xy/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
Python成长之路_装饰器的更多相关文章
- python基础16_闭包_装饰器
不了解是否其他语言也有类似 python 装饰器这样的东西. 最近才发现ECMAScript6也是有生成器函数的,也有 yield generator 装饰器的基础知识是闭包: # 闭包:嵌套函数, ...
- Python学习之路7☞装饰器
一:命名空间与作用域 1.1命名空间 局部命名空间: def foo(): x=1 def func(): pass 全局命名空间: import time class ClassName:pass ...
- Python学习之路6 - 装饰器
装饰器 定义:本质是函数,(装饰其他函数)就是为其他函数添加附加功能.原则:1.不能修改被装饰的函数的源代码 2.不能修改被装饰的函数的调用方式 实现装饰器的知识储备: 1.函数即“变量” 2.高阶函 ...
- (转)Python成长之路【第九篇】:Python基础之面向对象
一.三大编程范式 正本清源一:有人说,函数式编程就是用函数编程-->错误1 编程范式即编程的方法论,标识一种编程风格 大家学习了基本的Python语法后,大家就可以写Python代码了,然后每个 ...
- Python 标准库中的装饰器
题目描述 1.简单举例 Python 标准库中的装饰器 2.说说你用过的 Python 标准库中的装饰器 1. 首先,我们比较熟悉,也是比较常用的 Python 标准库提供的装饰器有:property ...
- 简学Python第四章__装饰器、迭代器、列表生成式
Python第四章__装饰器.迭代器 欢迎加入Linux_Python学习群 群号:478616847 目录: 列表生成式 生成器 迭代器 单层装饰器(无参) 多层装饰器(有参) 冒泡算法 代码开发 ...
- 初学 Python(十五)——装饰器
初学 Python(十五)--装饰器 初学 Python,主要整理一些学习到的知识点,这次是生成器. #-*- coding:utf-8 -*- import functools def curren ...
- 十一. Python基础(11)—补充: 作用域 & 装饰器
十一. Python基础(11)-补充: 作用域 & 装饰器 1 ● Python的作用域补遗 在C/C++等语言中, if语句等控制结构(control structure)会产生新的作用域 ...
- Python 函数修饰符(装饰器)的使用
Python 函数修饰符(装饰器)的使用 1. 修饰符的来源修饰符是一个很著名的设计模式,经常被用于有切面需求的场景,较为经典的有插入日志.性能测试.事务处理等. 修饰符是解决这类问题的绝佳设计, ...
随机推荐
- 【Lucene4.8教程之五】Luke
一.Luke基本内容 1.Luke简介 Luke可用于查看Lucene创建的索引,并对其进行基本操作. 2.创建Luke (1)从Github上下载源文件 https://github.com/tar ...
- 12个非常有用的JavaScript小技巧
在这篇文章中将给大家分享12个有关于JavaScript的小技巧.这些小技巧可能在你的实际工作中或许能帮助你解决一些问题. 使用!!操作符转换布尔值 有时候我们需要对一个变量查检其是否存在或者检查值是 ...
- 如何成为一个真正在路上的Linuxer
Linux 是工具,却更像一个信仰. 写在前面: 本文目的不是教你如何成为一个真正的Linuxer,也没有能力教你成为一个真正的linuxer,而是通过笔者的一些想法试图指引你真正踏上学习linux之 ...
- 学习http的一个网站
http://www.blogjava.net/zjusuyong/articles/304788.html
- webService设置超时时间
在客户端配置文件中设置: <bindings> <basicHttpBinding> <binding name="UrlCrawler ...
- SpringMVC原理+流程图
SpringMVC工作原理:整个处理过程从一个HTTP请求开始:1)DispatcherServlet接收到请求后,根据对应配置文件中配置的处理器映射,找到对应的处理器映射项(HandlerMappi ...
- table列等宽
固定table列等宽 固定table列等宽 style: {table-layout : fixed} html: style="table-layout:fixed" js: o ...
- SQL Server 2008空间数据应用系列十二:Bing Maps中呈现GeoRSS订阅的空间数据
原文:SQL Server 2008空间数据应用系列十二:Bing Maps中呈现GeoRSS订阅的空间数据 友情提示,您阅读本篇博文的先决条件如下: 1.本文示例基于Microsoft SQL Se ...
- Unix/Linux环境C编程入门教程(33) 命令和鼠标管理用户和组
Linux是一个多用户.多任务的实时操作系统,允许多人同时访问计算机, 并同时运行多个任务.UNIX系统具有稳定.高效.安全.方便.功能强大等诸多优点,自20世纪70年代开始便运行在许多大型和小型计算 ...
- HDOJ-1010 Tempter of the Bone(dfs+剪枝)
http://acm.hdu.edu.cn/showproblem.php?pid=1010 给出一个n*m的迷宫 X为墙 .为空地 S为起点 D为终点 给出时间T 每走一步花费一单位的时间 走过的空 ...