Python装饰器的使用

一、概念

装饰器是Python中独有的语法

  • 装饰器的概念:已知一个函数,如果需要给该函数增加新的功能,但是不希望修改原函数,在Python中,这种在代码运行期间动态执行的机制被称为装饰器【Decorator】

  • 装饰器的作用:为已经存在的函数或者类添加额外的功能

  • 装饰器的本质:实际上就是一个闭包

    • 闭包概念:内部函数访问外部函数中的变量【函数】
    • 闭包函数
    def outter(n):
     def inner():
         print(n)
     return inner
    f = outter(66)   # 调用outter函数, f ------>inner
    print(f)  # <function outter.<locals>.inner at 0x000001D6C0A6C3A0>
    f()  # 相当于调用inner函数
    

二、三种使用形式

2.1 使用一:直接使用

直接使用,较繁琐

  • 装饰器的使用
    • 1、定义闭包,给外部函数设置参数,参数表示需要被装饰的函数,命名:f/fun/func
    • 2、在内部函数中调用被装饰的函数,同时增加新的功能

需求:已有一个函数my_print,现在需要给他新增一个功能,多打印一条信息,但是不改变原函数

分析

  • 定义一个闭包函数
def outter(n):
 def inner():
     print(n)
 return inner
f = outter(66)   # 调用outter函数, f ------>inner
print(f)  # <function outter.<locals>.inner at 0x000001D6C0A6C3A0>
f()  # 相当于调用inner函数
  • 在闭包函数的内部函数中调用被装饰的函数,同时增加新的功能
# 已知函数
def my_print():
 print("拼搏到无能为力~~~~")

# 装饰器
# 1.定义闭包,给外部函数设置参数,参数表示需要被装饰的函数,命名:f/fun/func
def outter(func):
 def inner():
     # 2.在内部函数中调用被装饰的函数,同时增加新的功能
     print('inner:',func)  # inner: <function my_print at 0x00000147E9DEC430>
     func()   # 调用原函数my_print
     print('new~~~~~~~')  # 新增的功能
 return inner

f = outter(my_print)   # 需要装饰my_print,所以将my_print传参给func
f()        # 调用inner函数
  • 总结
    • outter是装饰器的名称【外部函数的函数名】
    • inner是装饰器的核心部分【调用原函数,新增新的功能】
    • 在inner中,新增功能还是调用原函数,理论上没有先后顺序,一般需要根据具体的需求决定

2.2 使用二:使用@xxx

使用说明:

  • 使用 @xxx 可以将一个装饰器作用于一个函数上,只需要将@xxx书写在一个函数的前面,则表示xxx装饰器装饰指定的函数
  • @xxx xxx表示装饰器的名称【外部函数的函数名】
  • 如果使用@xxx加载装饰器,则必须装饰器先存在,然后才能使用

代码演示

# 装饰器
def outter(func):
    def inner():
        print("inner:", func)
        func()
        print("new++++++")

    return inner


# 已知函数
@outter  # 等价于 f = outter(my_print),调用了outter()函数,同时将my_print传参给func
def my_print():
    print("拼搏到无能为力")


print(my_print)  # 此时不在指向原函数,<function outter.<locals>.inner at 0x0000019EE45AE480>指向了内部函数inner()
my_print()  # 等价于f(),实际调用了inner()函数
  • 装饰器函数写在原函数的前面
  • 在原函数的前面使用@outter,表示装饰器outter(),装饰了原函数 my_print()
    • 并且该代码等价于:第一种写法中的 f = outter(my_print) ,表示调用了 outter() 函数,同时将 my_print 传参给func
  • 此时打印my_print,可以发现,结果为<function outter.<locals>.inner at 0x0000019EE45AE480>
    • 不在指向了原函数,而是指向了内部函数inner()
  • 此时在调用my_print()
    • 该代码等价于:第一种写法中的f() ,表示调用了inner()函数,

2.3 使用三:一个装饰器装饰多个函数

需求:书写一个装饰器,同时装饰多个函数,给多个函数同时增加一个新的功能

  • 注意事项
    • 如果同一个装饰器装饰多个不同的函数,为了适配所有的函数,则给装饰器的内部函数设置不定长参数
# 装饰器
def outter(func):
    def inner(*args, **kwargs):
        func(*args, **kwargs)  # 调用原函数,注意传参问题
        print("new++++++")

    return inner

# 函数1
@outter
def a():
    print("aaaaaaa")

# 函数2
@outter
def b(num1, num2):
    print("bbbbbbb", num1, num2)

# 函数3
@outter
def c(x, y, z, num):
    print("ccccccc", x, y, z, num)


# 调用函数
a()
b(1, 2)
c(1, 2, 3, 4)
  • 函数1,2,3的参数数目都不相同,但是都需要用同一个装饰器,所以为了适配,给装饰器的内部函数设置不定长参数*args, **kwargs,将原函数的参数进行打包,在内部函数中通过func调用原函数,在传入参数*args, **kwargs进行解包

2.4实例演示

需求:书写一个装饰器,可以统计任意一个函数的执行时间

import time


def wrapper(func):
    def get_time(*args, **kwargs):
        start_time = time.time()  # 开始的时间戳
        func(*args, **kwargs)
        end_time = time.time()  # 结束的时间戳
        return end_time - start_time

    return get_time


@wrapper
def check():
    for i in range(1000000):
        pass


r = check()
print(f"花费的时间为:{r}")
  • 16
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
装饰器Python中一种强大的功能,允许在不修改原始函数的情况下,向函数添加额外的功能。装饰器使用@符号将其应用于函数。 一个装饰器函数通常具有以下形式: ```python def decorator(func): def wrapper(*args, **kwargs): # 添加额外的功能 return func(*args, **kwargs) return wrapper ``` 其中,`decorator`是装饰器函数本身,`wrapper`是内部函数,用于包装原始函数,并添加额外的功能。 装饰器可以应用于函数,类甚至是类的方法。通过在被装饰的函数或方法前使用装饰器函数,可以将装饰器应用于该函数或方法。 例如,引用中的代码展示了一个简单的装饰器模板,引用中的代码展示了一个使用装饰器实现函数计时器的例子。在这个例子中,`timec`是装饰器函数,`wrapper`是内部函数用于计算函数执行的时间。被`timec`装饰的函数`con_add`和`join_add`会在执行前后输出运行时间。 引用中的代码展示了一个包含关键字的装饰器的例子。在这个例子中,`decorator`是装饰器函数,`wrapper`是内部函数用于在执行被装饰的函数前后输出时间。被`decorator`装饰的函数`f`接受两个位置参数和任意个关键字参数,并在执行前后输出相应的信息。 总之,装饰器Python中一种强大的功能,可以用于给函数或方法添加额外的功能,而不需要修改原始函数的代码。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [Python装饰器使用](https://blog.csdn.net/belong_to_you/article/details/111003741)[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* *3* [Python装饰器使用](https://blog.csdn.net/qq_43830639/article/details/95247941)[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
发出的红包

打赏作者

巧克力配酸奶

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值