python装饰器

装饰器是:输入函数,并返回函数,的函数。是一个高阶/高层函数。是调用函数的函数。其返回值必须是一个函数。装饰器主要有以下三种类型:

    1. 单层的装饰器,只能在函数前面加东西。因为返回的是原始函数。
    1. 双层的装饰器,可以在函数前后加东西。返回的是内层函数。内层函数可以对被装饰函数进行修改。并在被装饰函数前后进行其他操作。 最内层函数可以不用返回函数,也可以不用返回任何数值。它仅仅使用了被装饰函数的结果,并且变成了新的被装饰函数。
    1. 三层的装饰器,可以给原函数添加新参数。逐次返回内层函数。 最外面函数负责传入新参数; 中间层函数负责装饰,其输入是被装饰的函数, 输出是最内层函数; 最内层函数负责装饰的具体内容,其输入是"*args, **kw",代表任意参数。最内层函数里面调用被装饰函数,并且使用了最外层函数传入的参数。 返回,修改后的被装饰函数的输出值。 在使用时,需要在装饰器后面加上参数。可以是多个参数。

注意: 当装饰器的函数结构超过(包含)两层时,需要在最内层函数前一行添加"@functools.wraps(func)",用于修改最内层函数的函数名为被装饰函数的函数名。

python 复制代码
import functools

def decorator_1(func):
    """
    单层的装饰器,只能在函数前面加东西,无法改变被装饰函数的输出结果。因为返回的是原始函数。
    """
    print(f"使用了单层装饰器:decorator_1")
    print(f"正在执行函数: {func.__name__}")  # 调用了函数
    return func

def decorator_2_case_1(func):
    """
    双层装饰器,可以在被装饰函数的前面和后面加东西。  该装饰器实际上装饰的是内层函数,返回的是内层函数。
    内层函数调用了被装饰的函数,需要在其前面添加命令"@functools.wraps(func)", 用于修改内层函数的函数名与被装饰函数的函数名一致。
    内层函数的参数是"*args, **kw",代表任意参数,用于普适化的传递被装饰函数的参数。
    
    """
    def inner(*args, **kw):
        print(f"使用了双层装饰器:decorator_2_case_1")
        print(f"正在执行函数: {func.__name__}")  # 调用了函数
        func(*args, **kw)
        print("执行函数结束。")
    return inner

def decorator_2_case_2(func):
    """
    双层的装饰器,除了可以在被装饰函数前后添加东西之外,(理论上)也可以修改被装饰函数的返回值,包括数据格式。
    但是,这种处理方式不推荐,除非用于校正已有函数的错误。否则有悖装饰器的初衷,不如重新写一个函数。
    再重复一遍:双层装饰器,本质上是装饰的内层函数,返回的也是内层函数,只不过内层函数调用了被装饰函数。
    """
    def inner(*args, **kw):
        print(f"使用了双层装饰器:decorator_2_case_2")
        print(f"正在执行函数: {func.__name__}")  # 调用了函数
        raw_result = func(*args, **kw)
        new_result = raw_result + 100
        print("执行函数结束。比原始数值增加了100.")
        return new_result
    return inner

def decorator_3(delta,aString):
    """
    三层装饰器,相比以上单层和双层装饰器,主要新增功能是:在装饰时传入新参数。
    最外层用于传入新参数;
    中间层用于传入被装饰函数;
    最内层用于同时使用新参数和被装饰函数,对被装饰函数的结果进行重新修改。
    总结:在最外层添加参数,在最里层使用。本质上装饰的是最内层函数,所以需要在最内层函数前面添加"@functools.wraps(func)"。
    """
    def decorator_middle(func):
        @functools.wraps(func)
        def decorator_inner(*args, **kw):
            print(f"使用了三层装饰器:decorator_3")
            print(f"正在执行函数: {func.__name__}。新增参数: {delta},{aString}")  # 调用了函数
            result = func(*args, **kw)
            new_result = result + delta
            print(f"执行函数结束。比原始数值增加了 {delta}.")
            return new_result
        return decorator_inner
    return decorator_middle

@ decorator_1
# @ decorator_2_case_1
# @ decorator_2_case_2
# @ decorator_3(88,"kkk")
def add(a,b):
    return a+b




if __name__ == "__main__":
    result = add(5,6)
    # print(result)
    # print("add.__name__ = ",add.__name__)

参考:

装饰器 - Python教程 - 廖雪峰的官方网站

相关推荐
fmdpenny26 分钟前
Vue3初学之商品的增,删,改功能
开发语言·javascript·vue.js
通信.萌新33 分钟前
OpenCV边沿检测(Python版)
人工智能·python·opencv
Bran_Liu38 分钟前
【LeetCode 刷题】字符串-字符串匹配(KMP)
python·算法·leetcode
涛ing41 分钟前
21. C语言 `typedef`:类型重命名
linux·c语言·开发语言·c++·vscode·算法·visual studio
weixin_3077791341 分钟前
分析一个深度学习项目并设计算法和用PyTorch实现的方法和步骤
人工智能·pytorch·python
等一场春雨1 小时前
Java设计模式 十四 行为型模式 (Behavioral Patterns)
java·开发语言·设计模式
黄金小码农1 小时前
C语言二级 2025/1/20 周一
c语言·开发语言·算法
萧若岚1 小时前
Elixir语言的Web开发
开发语言·后端·golang
wave_sky2 小时前
解决使用code命令时的bash: code: command not found问题
开发语言·bash
Channing Lewis2 小时前
flask实现重启后需要重新输入用户名而避免浏览器使用之前已经记录的用户名
后端·python·flask