欢迎添加华为云小助手微信(微信号:HWCloud002HWCloud003),输入关键字“加群”,加入华为云线上技术讨论群;输入关键字“最新活动”,获取华为云最新特惠促销。华为云诸多技术大咖、特惠活动等你来撩

关于诉求

昨天简单聊了下Flask的学习感想,然后分享了一些Flask的学习方式与视频。其中提到在学习Python Web前,请先将python基础及装饰器等知识有一个了解,这样学习起来不至于太过吃力。


然后,今天有朋友私信说对python的类和装饰器不甚了解,希望能讲讲这些知识。关于函数、方法、类,我之前发过一篇文章,就不再赘述了。其实去年详细总结过一篇关于Python装饰器的文章,只不过是在公司博客写的,没办法复制出来,所以今天就这之前的知识做一个总结和复习吧。

引子

谈及python装饰器,很多人第一时间想到的是@这个符号。在方法上一行加上@decorator就行了。但很多人看着都会、一用就跪,而且很多时候,我们都不知道什么场景下适合使用装饰器。那么今天就带大家一步步了解装饰器的使用吧

装饰器(Decorator)是python的一个重要部分,简单来说,他是修改其他函数功能的函数。

他们有助于让我们的代码更简短,也更Pythonic!

万物皆对象

在Python的世界中,万物皆对象,听起来比较抽象,但其实理解起来很简单,你可以用将任何一个变量、函数、方法、类等等赋值给另一个变量。只有你了解了这些,才能进一步的理解装饰器。

函数赋值

而学习装饰器前,我们先来举一个简单的函数例子:

  1. def hello(name='BreezePython'):
  2. return "say hello to {}".format(name)
  3. hello()
  4. # 'say hello to BreezePython'
  5. hi = hello
  6. hi('tommorow')
  7. # 'say hello to tommorow'

我们首先创建一个hello的函数,然后调用,之后将hello赋值给hi,之后调用hi,完成了相同的操作,也许你觉得这个例子so easy,那就接着往下走。

函数嵌套

当我们在函数中再次定义一个函数是,即完成了一个函数的嵌套操作:

  1. def hello():
  2. print('is hello function...')
  3. def hi():
  4. return 'hi ,nice to meet you!'
  5. def bye():
  6. return 'bye,stranger...'
  7. print(hi())
  8. print(bye())
  9. hello()
  10. output:
  11. is hello function...
  12. hi ,nice to meet you!
  13. bye,stranger...
  14. hi()
  15. output:
  16. Traceback (most recent call last):
  17. File "<input>", line 1, in <module>
  18. NameError: name 'hi' is not defined

上面的例子看到了,函数中可以调用子函数,但如果你直接去调用子函数,则会抛出未定义的异常,那么我们如何调用子函数?

函数中返回函数

让我们由浅入深,先考虑从函数中返回函数

  1. def hello(name=None):
  2. print('is hello function...')
  3. def hi():
  4. return 'hi ,nice to meet you!'
  5. def bye():
  6. return 'bye,stranger...'
  7. if name == 'BreezePython':
  8. return hi
  9. else:
  10. return bye
  11. main = hello('BreezePython')
  12. >>> output: is hello function...
  13. main()
  14. >>> output: 'hi ,nice to meet you!'

初学者可能对python中的小括号有些迷糊,添加小括号与否有什么影响呢?

当你在函数或者实例化的对象后添加小括号,代表立即执行;

然而,当你不添加小阔爱好时,他可以被到处传递,并可以复制给变得变量而不去执行它。

函数作为参数传递

  1. def child():
  2. return 'is child function...'
  3. def main(func):
  4. print('is main function...')
  5. print(func())
  6. main(child)
  7. output:
  8. >>> is main function...
  9. >>> is child function...

我们新建了一个child函数,然后将child话术传递给main函数,在main函数中调用child函数,达到了将函数作为参数传递的结果。

python闭包

我们先将上面的函数嵌套与传参来进行一下合并:

  1. def main(func):
  2. print('is main function...')
  3. def child():
  4. print('it print before exec func...')
  5. func()
  6. print('it print after exec func...')
  7. return child
  8. def alone():
  9. print("I'm alone function ...")
  10. fix = main(alone)
  11. fix()
  12. output:
  13. >>> is main function...
  14. >>> it print before exec func...
  15. >>> I'm alone function ...
  16. >>> it print after exec func...

通过合并,我们将上面两个例子进行和组装,变成了一个升级版的闭包,很多人会说,什么是闭包呢?其实很简单…

我们可以将闭包理解为一种特殊的函数,这种函数由两个函数的嵌套组成,

且称之为外函数和内函数,外函数返回值是内函数的引用,此时就构成了闭包。

first Decorator

上面的例子中,我们看到了一个闭包与函数传参的例子,那么装饰器是什么?其实就是闭包+函数传参,如果上面的例子你看懂了,那么现在你只需要对代码格式稍作修改,就变成了一个装饰器!

  1. def main(func):
  2. print('is main function...')
  3. def child():
  4. print('it print before exec func...')
  5. func()
  6. print('it print after exec func...')
  7. return child
  8. @main
  9. def alone():
  10. print("I'm alone function ...")
  11. alone()
  12. # output:
  13. >>> is main function...
  14. >>> it print before exec func...
  15. >>> I'm alone function ...
  16. >>> alone
  17. >>> it print after exec func...

我们只是修改了一行代码的格式,就转化成了装饰器,@main就代表func = main(func)

可这样就算完美的装饰器了么?NO….

我们将alone函数稍作变更,即可看出问题所在:

  1. def alone():
  2. print("I'm alone function ...")
  3. print(alone.__name__)

正常情况下,调用alone带引的alone.__name__就是函数名即alone,但如果我们是通过装其实调用后打印呢,结果是什么?相信大家能猜到,是child。child是main函数的内建函数,它重写了我们的函数名,如何解决这个问题呢?

使用functools.wraps

  1. from functools import wraps
  2. # first Decorator
  3. def main(func):
  4. print('is main function...')
  5. @wraps(func)
  6. def child():
  7. print('it print before exec func...')
  8. func()
  9. print('it print after exec func...')
  10. return child
  11. @main
  12. def alone():
  13. print("I'm alone function ...")
  14. print(alone.__name__)
  15. alone()
  16. # output:
  17. >>> is main function...
  18. >>> it print before exec func...
  19. >>> I'm alone function ...
  20. >>> alone
  21. >>> it print after exec func...

我们通过引入functools方法中的wraps,保证了函数名称的原始性

@wraps接受一个函数,进行装饰,并加入了复制函数名称、注释文档、参数列表等功能,这样可以是我们在装饰器里面访问在装饰之前的函数的属性

装饰器实例

装饰器比大量的使用在Flask、Django中,学好了它不管是对于你理解flask的路由,还是之后的代码开发都有很多帮助,那么我们来做个简单的例子,日志打印装饰器:

  1. import time
  2. from functools import wraps
  3. def log_level(level='DEBUG'):
  4. def log_format(func):
  5. @wraps(func)
  6. def format(*args, **kwargs):
  7. logtime = time.strftime('%Y-%m-%d %H:%M:%S', time.localtime())
  8. print("[{}]{}: ".format(level, logtime), end='')
  9. return func(*args, **kwargs)
  10. return format
  11. return log_format
  12. @log_level()
  13. def log1():
  14. print("Hello,Welcome to 清风Python...")
  15. @log_level('ERROR')
  16. def log2():
  17. print("清风Python 是我的公众号...")
  18. log1()
  19. time.sleep(1)
  20. log2()
  21. # output:
  22. >>> [DEBUG]2019-08-20 01:24:23: Hello,Welcome to 清风Python...
  23. >>> [ERROR]2019-08-20 01:24:24: 清风Python 是我的公众号...

类的装饰器

讲了这么多,本来觉得该结束了,可总觉得还差点什么!没错,我们只是讲到了函数的装饰器,那么类的装饰器该如何操作呢?

  1. import time
  2. from functools import wraps
  3. class Logger:
  4. def __init__(self,level='DEBUG'):
  5. self.level = level
  6. def __call__(self, func):
  7. @wraps(func)
  8. def log_format(*args, **kwargs):
  9. log_time = time.strftime('%Y-%m-%d %H:%M:%S', time.localtime())
  10. print("[{}]{}: ".format(self.level, log_time), end='')
  11. return func(*args, **kwargs)
  12. return log_format
  13. @Logger()
  14. def log1():
  15. print("Hello,Welcome to 清风Python...")
  16. @Logger('Error')
  17. def log2():
  18. print("清风Python 是我的公众号...")
  19. log1()
  20. time.sleep(1)
  21. log2()
  22. # output:
  23. >>> [DEBUG]2019-08-20 01:24:23: Hello,Welcome to 清风Python...
  24. >>> [ERROR]2019-08-20 01:24:24: 清风Python 是我的公众号...

今天的装饰器内容就分享到这里吧…

The End

OK,今天的内容就到这里,如果觉得内容对你有所帮助,欢迎点击文章右下角的“在看”。

期待你关注我的公众号清风Python,如果觉得不错,希望能动动手指转发给你身边的朋友们。

作者:清风Python

Python装饰器总结,带你几步跨越此坑!的更多相关文章

  1. [python 基础]python装饰器(二)带参数的装饰器以及inspect.getcallargs分析

    带参数的装饰器理解无非记住两点: 1.本质不过在基本的装饰器外面再封装一层带参数的函数 2.在使用装饰器语法糖的时候与普通装饰器不同,必须要加()调用,且()内的内容可以省略(当省略时,admin默认 ...

  2. Python装饰器实现带参数和不带参数

    1 def log(text=None): 2 3 if isinstance(text, str): 4 def decorator(func): 5 @functools.wraps(func) ...

  3. 【转】九步学习python装饰器

    本篇日志来自:http://www.cnblogs.com/rhcad/archive/2011/12/21/2295507.html 纯转,只字未改.只是为了学习一下装饰器.其实现在也是没有太看明白 ...

  4. Python高级特性: 12步轻松搞定Python装饰器

    12步轻松搞定Python装饰器 通过 Python 装饰器实现DRY(不重复代码)原则:  http://python.jobbole.com/84151/   基本上一开始很难搞定python的装 ...

  5. python 装饰器 (多个参数的函数,带参数的装饰器)

    最简单的模板是这样的 #-*-coding:utf-8-*- def outer(func): def inner(): print 'before' func() print 'after' # r ...

  6. Python 装饰器学习

    Python装饰器学习(九步入门)   这是在Python学习小组上介绍的内容,现学现卖.多练习是好的学习方式. 第一步:最简单的函数,准备附加额外功能 1 2 3 4 5 6 7 8 # -*- c ...

  7. (转载)Python装饰器学习

    转载出处:http://www.cnblogs.com/rhcad/archive/2011/12/21/2295507.html 这是在Python学习小组上介绍的内容,现学现卖.多练习是好的学习方 ...

  8. Python装饰器学习

    Python装饰器学习(九步入门)   这是在Python学习小组上介绍的内容,现学现卖.多练习是好的学习方式. 第一步:最简单的函数,准备附加额外功能 ? 1 2 3 4 5 6 7 8 # -*- ...

  9. 自创最精简的python装饰器

    个人心血原创,欢迎转载,请注明作者和出处.否则依法追究法律责任!!!! author:headsen  chen date:2018-03-21  10:37:52 代码: 代码解析过程:1,def ...

随机推荐

  1. 基于xposed Hook框架实现个人免签支付方案

    我的个人网站如何实现支付功能? 想必很多程序员都有过想开发一个自己的网站来获得一些额外的收入,但做这件事会遇到支付这个问题.目前个人网站是无法实现支付功能的. 今天我就给大家分享一下我的实现方案:&l ...

  2. 『题解』Codeforces446C DZY Loves Fibonacci Numbers

    更好的阅读体验 Portal Portal1: Codeforces Portal2: Luogu Description In mathematical terms, the sequence \( ...

  3. 1114作业 html

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  4. python面试看这一篇就够了

    python-面试通关宝典 有面Python开发方向的,看这一个repo就够啦? 语言特性 1.谈谈对 Python 和其他语言的区别 Python属于解释型语言,当程序运行时,是一行一行的解释,并运 ...

  5. linux下安装opencv3.0

    查版本gcc --version 需>4.8python 2.7+cmake --version numpy 以上是必须的 linux下安装opencv3.0<pre>https:/ ...

  6. docker项目——搭建飞机大战小游戏

    项目2:搭建打飞机小游戏,验证数据持久化(最底下有链接) 第一步:拉取镜像 [root@localhost docker-image]# docker load < httpd_img.tar. ...

  7. nyoj 34-韩信点兵(暴力)

    34-韩信点兵 内存限制:64MB 时间限制:3000ms Special Judge: No accepted:34 submit:41 题目描述: 相传韩信才智过人,从不直接清点自己军队的人数,只 ...

  8. selenium介绍和环境搭建

    一.简单介绍 1.selenium:Selenium是一个用于Web应用程序测试的工具.Selenium测试直接运行在浏览器中,就像真正的用户在操作一样.支持的浏览器包括IE.Mozilla Fire ...

  9. Python第五天 列表练习 元组类型 字典类型 小购物车练习

    # 反转 reverse# l=['lili','asdf','qwer','80000']# l.reverse()# print(l) # ['80000', 'qwer', 'asdf', 'l ...

  10. Mac下载ChromeDriver

    ChromeDriver下载地址: https://npm.taobao.org/mirrors/chromedriver 如何查看chrome版本与ChromeDriver版本对应 查看chrome ...