Python装饰器

一. 有时候我们会有这样需求: 在原有的逻辑前后添加一段逻辑

如: 在增/删/改操作之前检查用户是否登录、某个操作前后添加一些log ...

在java中你只能重写源码. 例如有一个视频上传的方法:
public void upload(Video video) { ... }
调用处: upload(video);
现在你需要在视频上传方法调用时记录一条log.
这时你可能会这样处理:

public void uploadVideoWithLog(Video video) {
  System.out.println("upload method execute !");
  upload(video);
}

调用之处得改成: uploadVideoWithLog(video);
可以看出, 这对原有代码逻辑的改动是非常大的.

为了灵活性和减少对原有代码的破坏性, 你可能会使用spring的AOP或者使用AspectJ. 对于Python 来说有更简单的处理方法, 即使用装饰器.

二. Python装饰器 (Python在原有逻辑前后添加逻辑的手段)

  1. 不用装饰器的处理方式.
    #原有函数
    def foo():
    print('I am foo !')

     # 重新定义一个函数, 包装原有函数
     def use_log(func):
         # 需要添加的逻辑
         print('function %s() executing ' % func.__name__)
         func() #真正工作委托给原有函数
    
     # 调用函数 (原来是这样调用: foo())
     use_log(foo)
    

输出:

function foo() execute 
I am foo !
  1. 使用装饰器.
    # 定义需要添加的逻辑 (定义装饰器)
    def log(func):
    def decorator(*args, *kwargs):
    print('function %s() execute ' % func.name)
    return func(
    args, **kwargs)
    return decorator

     @log # 在原有函数上应用装饰逻辑
     def foo(): #原有函数
         print('I am foo !') 
    
     #调用函数 (和原有的调用逻辑一致, 不需要更改)
     foo()
    

输出:

function foo() execute 
I am foo !
  1. 可见装饰器对于原有代码的破坏性要小的多, 它只需要定义要添加的逻辑, 并在原有函数之前添加@decorator即可, 调用逻辑不用改变. 这是一个非常好的应用开闭原则的例子 (对更改关闭, 对扩展开发. 添加逻辑就是属于扩展).

三. functools.wraps

  1. 在原有代码之上应用装饰逻辑 (即在原来函数之前添加@decorator), 其实是Python内置的语法糖.
    def decorator(func):
        def delegate(*args, **kwargs):
            print('function %s() execute ' % func.__name__)
            return func(*args, **kwargs)
    
        return delegate
    
    # 在一个函数(我们叫暂且他目标函数吧, target_func)上添加装饰逻辑(decorator函数),   
    # 等价于: target_func = decorator(target_func)
    # 即调用装饰函数(以目标函数为参数)生成一个新函数替换原有的目标函数
    @decorator
    def foo():
        print('I am foo !')

上面这段代码其实等价于:

    def decorator(func):
        def delegate(*args, **kwargs):
            print('function %s() execute ' % func.__name__)
            return func(*args, **kwargs)
    
        return delegate
        
    def foo():
        print('I am foo !')
    
    foo = decorator(foo)

添加@decorator其实就是把函数的内容重新赋值了 (以目标函数为参数, 调用decorator函数重新生成一个新的函数, 用新生成的函数替换目标函数). 这样下来目标函数的属性就变了.
看看面的程序会输出什么 ?
def decorator(func):
def delegate(*args, *kwargs):
print('function %s() execute ' % func.name)
return func(
args, **kwargs)
return delegate

    #下面的调用等价于: foo = decorator(foo)
    @decorator 
    def foo():
        """description of function foo !!!!!!!!!"""
        print('I am foo !')

    print(foo.__name__) #输出: delegate
    print(foo.__doc__)  #输出: None

输出:

delegate
None

可以看到foo()函数的属性确实变了. 因为调用decorator()函数生成的函数是delegate()函数, 因此__name__属性的值是delegate; 而delegate()函数并没有定义函数文档, 因此__doc__属性值为None.
那么, 如何在添加装饰器之后还能保持原有函数的属性呢 ??

  1. functools.wraps (指定某个函数的属性)
    可以使用python内置的functools.wraps装饰器来解决上面的问题.
    from functools import wraps
    def decorator(func):
    @wraps(func) #让下面的函数使用func函数的属性
    def delegate(*args, *kwargs):
    print('function %s() execute ' % func.name)
    return func(
    args, **kwargs)
    return delegate

     #下面的调用等价于: foo = decorator(foo)
     @decorator 
     def foo():
         """description of function foo !!!!!!!!!"""
         print('I am foo !')
    
     print(foo.__name__) #输出: foo
     print(foo.__doc__)  #输出: description of method foo !!!!!!!!!
    

输出:

foo
description of method foo !!!!!!!!!

这样, 在应用了装饰器之后foo()函数仍能保持其原有属性.
functools.wraps就是一个装饰器. 最简单的用法就是改变一个函数的属性. 如下, 将foo()函数的属性改为函数bar()的属性:
from functools import wraps

    def bar():
        """description of function bar !!!!!!"""
        pass
    
    @wraps(bar)
    def foo():
        pass
    
    print(foo.__name__) #输出: bar
    print(foo.__doc__)  #输出: description of function bar !!!!!!

tips: 上述代码环执行境为Python 3.5.1`, 所使用的IDE为PyCharm

.
.
错误之处在所难免, 欢迎指正~~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,482评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,377评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,762评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,273评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,289评论 5 373
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,046评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,351评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,988评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,476评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,948评论 2 324
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,064评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,712评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,261评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,264评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,486评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,511评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,802评论 2 345

推荐阅读更多精彩内容

  • 每个人都有的内裤主要功能是用来遮羞,但是到了冬天它没法为我们防风御寒,咋办?我们想到的一个办法就是把内裤改造一下,...
    chen_000阅读 1,360评论 0 3
  • Python装饰器的高级用法(翻译) 原文地址https://www.codementor.io/python/t...
    城南道阅读 4,805评论 1 22
  • www.yunxcloud.cn 首先要明白装饰器是用来给函数增加额外功能的。 常用的工具函数 import ti...
    彩色系阅读 1,011评论 0 1
  • 呵呵!作为一名教python的老师,我发现学生们基本上一开始很难搞定python的装饰器,也许因为装饰器确实很难懂...
    TypingQuietly阅读 19,518评论 26 186
  • 我生活的城市,没有pageone,没有字里行间,没有方所,没有先锋书店。最有名的书店叫新华书店。 新华书店大概从我...
    勇猛精进的王糟糟阅读 246评论 0 0