Python全栈开发
Python全栈开发
一文让你彻底明白Python装饰器原理,从此面试工作再也不怕了。
一、装饰器
装饰器可以使函数执行前和执行后分别执行其他的附加功能,这种在代码运行期间动态增加功能的方式,称之为“装饰器”(Decorator),装饰器的功能非常强大,但是理解起来有些困难,因此我尽量用最简单的例子一步步的说明这个原理。
1、不带参数的装饰器
假设我定义了一个函数f,想要在不改变原来函数定义的情况下,在函数运行前打印出start,函数运行后打印出end,要实现这样一个功能该怎么实现?看下面如何用一个简单的装饰器来实现:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
# 使用@语法放在函数的定义上面 相当于执行 f=outer(f),此时f赋值成为了一个新的outer函数,# 此时f函数就指向了outer函数的返回值inner,inner是一个函数名,定义在oute函数里面# 原来的f是函数名可简单理解为一个变量,作为outer函数的参数传递进去了 此时参数func相当于fdef outer(func): # 定义一个outer函数作为装饰器 def inner(): # 如果执行inner()函数的话步骤如下: print('start') # 1、首先打印了字符‘start’, r=func() # 2、执行func函数,func函数相当于def f(): print('中') print('end') # 3、接着函数打印‘end’ return r # 4、将func函数的结果返回 return inner@outerdef f(): # f=outer(f)=innner print('中')f() # f()相当于inner(),执行inner函数的步骤看上面定义处的注释<br>#打印结果顺序为 start 中 end |
2、包含任意参数的装饰器
在实际中,我们的装饰器可能应用到不同的函数中去,这些函数的参数都不一样,那么我们怎么实现一个对任意参数都能实现功能的装饰器?还记得我写函数那篇博客中,就写一种可以接受任意参数的函数,下面来看看如何将其应用到装饰器中去
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
#其实只要将上面一种不带参数的装饰器修改一下就可以了#修改也很简单,只需将inner和func的参数改为 (*args,**kwargs)#其他实现的过程和上面一种一样,就不再介绍了def outer(func): def inner(*args,**kwargs): print('start') r=func(*args,**kwargs) # 这里func(*args,**kwargs)相当于f(a,b) print('end') return r return inner@outerdef f(a,b): print(a+b)f(1,4) # f(1,4)相当于inner(1,4) 这里打印的结果为 start 5 end |
3、使用两个装饰器
当一个装饰器不够用的话,我们就可以用两个装饰器,当然理解起来也就更复杂了,当使用两个装饰器的话,首先将函数与内层装饰器结合然后在与外层装饰器相结合,要理解使用@语法的时候到底执行了什么,是理解装饰器的关键。这里还是用最简单的例子来进行说明。
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
def outer2(func2): def inner2(*args,**kwargs): print('开始') r=func2(*args,**kwargs) print('结束') return r return inner2def outer1(func1): def inner1(*args,**kwargs): print('start') r=func1(*args,**kwargs) print('end') return r return inner1@outer2 # 这里相当于执行了 f=outer1(f) f=outer2(f),步骤如下@outer1 #1、f=outer1(f) f被重新赋值为outer1(1)的返回值inner1,def f(): # 此时func1为 f():print('f 函数') print('f 函数') #2、f=outer2(f) 类似f=outer2(inner1) f被重新赋值为outer2的返回值inner2 # 此时func2 为inner1函数 inner1里面func1函数为原来的 f():print('f 函数') f() # 相当于执行 outer2(inner1)()>>开始 # 在outer函数里面执行,首先打印 ‘开始 ’>>start # 执行func2 即执行inner1函数 打印 ‘start’>>f 函数 # 在inner1函数里面执行 func1 即f()函数,打印 ‘f 函数’>>end # f函数执行完,接着执行inner1函数里面的 print('end')>>结束 # 最后执行inner2函数里面的 print('结束') |
4、带参数的装饰器
前面的装饰器本身没有带参数,如果要写一个带参数的装饰器怎么办,那么我们就需要写一个三层的装饰器,而且前面写的装饰器都不太规范,下面来写一个比较规范带参数的装饰器,下面来看一下代码,大家可以将下面的代码自我运行一下
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
import functoolsdef log(k=''): #这里参数定义的是一个默认参数,如果没有传入参数,默认为空,可以换成其他类型的参数 def decorator(func): @functools.wraps(func) #这一句的功能是使被装饰器装饰的函数的函数名不被改变, def wrapper(*args, **kwargs): print('start') print('{}:{}'.format(k, func.__name__)) #这里使用了装饰器的参数k r = func(*args, **kwargs) print('end') return r return wrapper return decorator@log() # fun1=log()(fun1) 装饰器没有使用参数def fun1(a): print(a + 10)fun1(10)# print(fun1.__name__) # 上面装饰器如果没有@functools.wraps(func)一句的话,这里打印出的函数名为wrapper@log('excute') # fun2=log('excute')(fun2) 装饰器使用给定参数def fun2(a): print(a + 20)fun2(10) |
Python全栈开发的更多相关文章
- Python全栈开发【面向对象进阶】
Python全栈开发[面向对象进阶] 本节内容: isinstance(obj,cls)和issubclass(sub,super) 反射 __setattr__,__delattr__,__geta ...
- Python全栈开发【面向对象】
Python全栈开发[面向对象] 本节内容: 三大编程范式 面向对象设计与面向对象编程 类和对象 静态属性.类方法.静态方法 类组合 继承 多态 封装 三大编程范式 三大编程范式: 1.面向过程编程 ...
- Python全栈开发【模块】
Python全栈开发[模块] 本节内容: 模块介绍 time random os sys json & picle shelve XML hashlib ConfigParser loggin ...
- Python全栈开发【基础四】
Python全栈开发[基础四] 本节内容: 匿名函数(lambda) 函数式编程(map,filter,reduce) 文件处理 迭代器 三元表达式 列表解析与生成器表达式 生成器 匿名函数 lamb ...
- Python全栈开发【基础三】
Python全栈开发[基础三] 本节内容: 函数(全局与局部变量) 递归 内置函数 函数 一.定义和使用 函数最重要的是减少代码的重用性和增强代码可读性 def 函数名(参数): ... 函数体 . ...
- Python全栈开发【基础二】
Python全栈开发[基础二] 本节内容: Python 运算符(算术运算.比较运算.赋值运算.逻辑运算.成员运算) 基本数据类型(数字.布尔值.字符串.列表.元组.字典) 其他(编码,range,f ...
- Python全栈开发【基础一】
Python全栈开发[第一篇] 本节内容: Python 的种类 Python 的环境 Python 入门(解释器.编码.变量.input输入.if流程控制与缩进.while循环) if流程控制与wh ...
- python 全栈开发之路 day1
python 全栈开发之路 day1 本节内容 计算机发展介绍 计算机硬件组成 计算机基本原理 计算机 计算机(computer)俗称电脑,是一种用于高速计算的电子计算机器,可以进行数值计算,又可 ...
- 老男孩最新Python全栈开发视频教程(92天全)重点内容梳理笔记 看完就是全栈开发工程师
为什么要写这个系列博客呢? 说来讽刺,91年生人的我,同龄人大多有一份事业,或者有一个家庭了.而我,念了次985大学,年少轻狂,在大学期间迷信创业,觉得大学里的许多课程如同吃翔一样学了几乎一辈子都用不 ...
随机推荐
- wiki oi 3115高精度练习之减法
题目描述 Description 给出两个正整数A和B,计算A-B的值.保证A和B的位数不超过500位. 输入描述 Input Description 读入两个用空格隔开的正整数 输出描述 Outpu ...
- Android网络开发之用tcpdump抓包
Android开发过程中,当涉及到网络通信的时候,有一些字段须要抓包获取.我之前由于SSDP设备发现的包头格式没有写对,经过抓包分析和标准包头对照发现了这个困扰我非常久的问题.总之,掌握在Androi ...
- Codeforces 41D Pawn 简单dp
题目链接:点击打开链接 给定n*m 的矩阵 常数k 以下一个n*m的矩阵,每一个位置由 0-9的一个整数表示 问: 从最后一行開始向上走到第一行使得路径上的和 % (k+1) == 0 每一个格子仅仅 ...
- Matrix Factorization, Algorithms, Applications, and Avaliable packages
矩阵分解 来源:http://www.cvchina.info/2011/09/05/matrix-factorization-jungle/ 美帝的有心人士收集了市面上的矩阵分解的差点儿全部算法和应 ...
- Codeforces Round #246 (Div. 2)
题目链接:Codeforces Round #246 (Div. 2) A:直接找满足的人数,然后整除3就是答案 B:开一个vis数组记录每一个衣服的主场和客场出现次数.然后输出的时候主场数量加上反复 ...
- ThinkPHP - 自定义扩展类库
首先在想要使用类库的地方建立文件夹,文件名称随意,不能使用class 然后在配置文件中: 'AUTOLOAD_NAMESPACE' => array( 'Lib' => './Lib', ...
- PHP - 继承 - 子类使用父类方法
<?php class ShopProduct { private $FirstName; private $LastName; private $Title; private $Price; ...
- [Swust OJ 1094]--中位数(巧用set,堆排序)
题目链接:http://acm.swust.edu.cn/problem/1094/ Time limit(ms): 1000 Memory limit(kb): 32768 中位数(又称中值,英 ...
- int_float_double数据类型的存储格式。
一段用来检测编辑器存储方式的程序 //date : 2013/8/16 //designer :pengxiaoen //function check the C programmable langu ...
- 年度酷工作---高级数据工程师(公司靠谱,技术强悍,产品牛叉,福利有干货) 关键词:7000万用户、五星级厨师、住房补助 - V2EX
年度酷工作---高级数据工程师(公司靠谱,技术强悍,产品牛叉,福利有干货) 关键词:7000万用户.五星级厨师.住房补助 - V2EX 年度酷工作---高级数据工程师(公司靠谱,技术强悍,产品牛叉,福 ...