python如何使用装饰器

python装饰器可以极大地方便我们在对于重复执行的代码却又不确定有多少方法重复执行时,大大简化任务

可以用在方法执行的前置条件,判断方法是不是要执行,比如权限

# -*- coding: utf-8 -*-
def w1(func):
    def inner():
        print("---正在验证权限---")
        if True:
            print("--有权限--")
        else:
            print("--无权限--")
        func()
    return inner

# @w1 等价于 f1 = w1(f1)
@w1
def f1():
    print("---f1---")

@w1
def f2():
    print("---f2---")


f1()
f2()

打印结果如下:

---正在验证权限---
--有权限--
---f1---
---正在验证权限---
--有权限--
---f2---

若是有多个装饰器,那么从最上面的装饰器开始,把其下面的装饰器和方法都当做一个整体进行执行

# -*- coding: utf-8 -*-

def zhuangShi1(func):
    def inner():
        print("---1---")
        return "第一次添加---" + func()
    return inner

def zhuangShi2(func):
    def inner():
        print("---2---")
        return "第二次添加---" + func()
    return inner

@zhuangShi1
@zhuangShi2
def f1():
    return "原始方法"

res = f1()
print(res)

打印结果如下:

---1---
---2---
第一次添加---第二次添加---原始方法

注意,如果有两层装饰器,并且在装饰器装饰时有执行代码,那么在装饰的时候,先走里面再走外面.在运行的时候,先运行外面载运行里面.
可以想象成身份证被包裹成邮件,包裹的时候,先看到身份证,最后再在外层包裹上写上地址信息.而拿到包裹时,先看到包裹上面的地址信息,拆开后才能看到身份信息

# -*- coding: utf-8 -*-

def zhuangShi1(func):
    print("正在装饰1")  # 这是在最外层包裹
    def inner():
        print("---1---")  # 这是外层执行
        return "第一次添加---" + func()
    return inner
          
def zhuangShi2(func):
    print("正在装饰2")  # 这是里层包裹
    def inner():   
        print("---2---")  # 这是里层执行
        return "第二次添加---" + func()
    return inner
          
@zhuangShi1        
@zhuangShi2        
def f1():          
    return "原始方法"
                   
res = f1()         
print(res)         

输出结果是:

正在装饰2
正在装饰1
---1---
---2---
第一次添加---第二次添加---原始方法

如果对于带有参数的装饰起来说,只需要在装饰器方法的闭包加参数,在闭包中调用方法时也加上参数即可

def zhuangShi(func):
    print("---不定长参数的装饰器---")
    def inner(*args, **kwargs): # 这里加上参数
        print("---inner---")
        func(*args, **kwargs) # 这里传入参数
    return inner

@zhuangShi
def func1(a, b, c):
    print("a = %d, b = %d, c = %d"%(a, b, c))

@zhuangShi
def func2(a, b, e):
    print("a = %d, b = %d, e = %d"%(a, b, e))


func1(1,2,3)
func2(11,22,e=33)

输出结果是:

---不定长参数的装饰器---
---不定长参数的装饰器---
---inner---
a = 1, b = 2, c = 3
---inner---
a = 11, b = 22, e = 33

如果函数带有返回值,那么只需要在闭包中把返回值保存,然后再在最后把得到的返回值返回即可

def zhuangShi(func):
    print("---不定长参数的装饰器---")
    def inner(*args, **kwargs):
        print("---inner---")
        res = func(*args, **kwargs)  # 这里将之前的函数返回值保存起来
        return res  # 把函数返回值返回
    return inner

@zhuangShi
def func1(a, b, c):
    print("a = %d, b = %d, c = %d"%(a, b, c))
    return "hahaha"


res = func1(1,2,3)
print("res is %s"%res)

输出结果为:

---不定长参数的装饰器---
---inner---
a = 1, b = 2, c = 3
res is hahaha


通用装饰器,适合有无返回值,有无参数,都可以

# 通用装饰器
def zhuangShi(func):
    def inner(*args, **kwargs):
        print("通用装饰器---打印日志")
        res = func(*args, **kwargs)
        return res
    return inner

@zhuangShi
def test1():
    print("-----test1----- 带返回值")
    return "test返回值"

@zhuangShi
def test2():
    print("-----test2----- 不带返回值")

@zhuangShi
def test3(a):
    print("-----test3----- 带参数,参数a = %d"%a)


res = test1()
print("test1 的返回值是: %s"%res)
test2()
test3(45)

输出结果:

通用装饰器---打印日志
-----test1----- 带返回值
test1 的返回值是: test返回值
通用装饰器---打印日志
-----test2----- 不带返回值
通用装饰器---打印日志
-----test3----- 带参数,参数a = 45

对于带参数的装饰器,则需要在寻常装饰其方法外再包一层函数

def zhuangShi(arg):  # 常见装饰器方法外再包一层  
    def zhuangShi_in(func):  # 装饰器方法
        def inner():
            print("--带有参数的装饰器--%s"%arg)
            func()
        return inner
    return zhuangShi_in  # 这里要返回装饰器方法

@zhuangShi("--哈哈哈哈哈--")
def test():
    print("test")

test()

输出结果

--带有参数的装饰器----哈哈哈哈哈--
test


类装饰器, 使用一个类来当做函数的装饰器, 要用到类的call( ) 方法,此方法可以使类能够像函数那样通过类名直接调用call方法

class Test(object):
    def __init__(self, func):
        print("--类装饰器初始化--")
        print("被装饰的方法名是: %s"%func.__name__)
        self.__func = func

    #__call__方法可以使类名能够像方法一样直接被调用
    def __call__(self):
        print("--装饰器中的功能--")
        self.__func()

@Test   #装饰器相当于 test = Test(test)
def test():
    print("-------test方法---------")


test()

输出结果是:

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,579评论 18 139
  • //Clojure入门教程: Clojure – Functional Programming for the J...
    葡萄喃喃呓语阅读 3,616评论 0 7
  • 第5章 引用类型(返回首页) 本章内容 使用对象 创建并操作数组 理解基本的JavaScript类型 使用基本类型...
    大学一百阅读 3,204评论 0 4
  • 喜欢一个人 简简单单 背起行囊 穿行于原始森林 溪水叮咚 蜂鸟欢唱 聆听大自然的 天籁之音 喜欢一个人 享受寂寞 ...
    远方孤雁阅读 150评论 0 1
  • 我现在还没解决这个问题,等我解决好了,我一定要好好记录一些这个问题,真心是我的困扰。 问题:我真机测试没有问题,但...
    为心而狂阅读 386评论 0 2