Python基础之装饰器

一:什么是装饰器

        器:指的是工具,可以定义为函数

        装饰:指的是为其他事物添加额外的东西点缀

        装饰器:指的是定义一个函数,该函数是用来给其他函数添加额外的功能

二、为什么要用装饰器

        开放封闭原则:

        开放:指的是对拓展功能是开放的

        封闭:指的是对修改源代码是封闭的

对扩展开放,意味着有新的需求或变化时,可以对现有代码进行扩展,以适应新的情况。对修改封闭,意味着对象一旦设计完成,就可以独立完成其工作,而不要对其进行修改。

装饰器就是在不修改被装饰对象源代码以及调用方式的前提下为被装饰对象添加新功能。

三、装饰器的实现

函数装饰器分为:无参装饰器和有参装饰两种,二者的实现原理一样,都是’函数嵌套+闭包+函数对象’的组合使用的产物。

3.1:无参装饰器的实现

无参装饰器的模版

def outer(func):
    def wrapper():
        res = func()
        return res
    return wrapper

@outer    # 加在被装饰对象的上方

如果想为下述函数添加统计其执行时间的功能

import time
def get_time():
    time.sleep(3)
    print("welcome")
    return 200
get_time()

使用装饰器:

import time

def outer(func):
    def wrapper():
        stat_time = time.time()
        res = func()
        end_time = time.time()
        print(f'程序运行的时间为{end_time-stat_time}')
        return res
    return wrapper

@outer
def get_time():
    time.sleep(3)
    print("welcome")
    return 200

get_time()

输出结果:

welcome
程序运行的时间为3.0009207725524902

3.2:有参装饰器的实现

有参装饰器的模版

def outer(func):
    def wrapper(*args, **kwargs):
        res = func(*args, **kwargs)
        return res
    return wrapper

@outer    # 加在被装饰对象的上方

如果想为下述函数添加统计其执行时间的功能

import time

def time_demo(a, b):
    time.sleep(3)
    print("welcome")
    print(a+b)

time_demo(3, 5)

使用装饰器:

import time

def outer(func):
    def wrapper(a, b):
        stat_time = time.time()
        res = func(a, b)
        end_time = time.time()
        print(f'程序运行的时间为{end_time-stat_time}')
        return res
    return wrapper

@outer
def time_demo(a, b):
    time.sleep(3)
    print("welcome")
    print(a+b)

time_demo(3, 5)

输出结果:

welcome
8
程序运行的时间为3.0009207725524902

3.3:装饰器函数本身带有参数

@logging(level='INFO')它就是一个函数,会被立刻执行,返回结果是一个装饰器。

与上面的例子相同:

import time

def loglevel(level):
    def outer(func):
        def wrapper(a, b):
            stat_time = time.time()
            res = func(a, b)
            end_time = time.time()
            print(f'程序运行的时间为{end_time-stat_time}')
            if level == 'INFO':
                print("level is INFO")
            return res
        return wrapper
    return outer

@loglevel(level='INFO')    # 相当于:time_demo=loglevel("INFO")(time_demo), -> loglevel("INFO") = outer, 因此time_demo=outer(time_demo)
def time_demo(a, b):
    time.sleep(3)
    print("welcome")
    print(a+b)

time_demo(3, 5)

@loglevel(level='INFO'):先执行loglevel(level='INFO'),得到outer返回值后,再执行@outer

输出结果:

welcome
8
程序运行的时间为3.0051047801971436
level is INFO

3.4:叠加多个装饰器

@deco3
@deco2
@deco1
def index():
    pass

执行顺序是:从最上面的装饰器依次往下执行,将下面的作为参数传入

index=deco3(deco2(deco1(index)))
  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Python中,装饰器是一种用于修改函数行为的函数。而带参数的装饰器就是在普通装饰器基础上,给装饰器传递额外的参数。这样可以在装饰器内部使用这些参数来改变装饰函数的行为。 在给装饰器传递参数时,需要在装饰器函数上面再包裹一层函数来接收参数,并返回装饰器函数。例如,示例代码中的@wrapper_out('QQ')就是一个带参数的装饰器。 当函数执行到带参数装饰器@wrapper_out('QQ')这句时,会先执行wrapper_out('QQ')这个函数,将参数'QQ'传给parameter并得到返回值wrapper函数。然后,再将@符号与wrapper函数结合,得到一个标准的装饰器,按照装饰器的执行流程来装饰函数。 通过带参数的装饰器,可以在装饰器内部对被装饰函数进行进一步的处理或设置,从而实现更灵活的函数装饰。这对于需要根据不同的参数来选择不同装饰方式的情况非常有用。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [python中带有参数的装饰器](https://blog.csdn.net/weixin_44799217/article/details/118695357)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] - *2* [python如何修改装饰器中参数](https://download.csdn.net/download/weixin_38570854/12870003)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值