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教程 - 廖雪峰的官方网站

相关推荐
JavaEdge在掘金3 分钟前
启动nginx报错,80 failed (97: Address family not supported by protocol)
python
纪元A梦10 分钟前
华为OD机试真题——绘图机器(2025A卷:100分)Java/python/JavaScript/C++/C/GO最佳实现
java·javascript·c++·python·华为od·go·华为od机试题
钢铁男儿13 分钟前
C# 深入理解类:面向对象编程的核心数据结构
开发语言·数据结构·c#
程序员小远22 分钟前
接口测试和单元测试详解
自动化测试·软件测试·python·测试工具·单元测试·测试用例·接口测试
Tech Synapse31 分钟前
电商商品推荐系统实战:基于TensorFlow Recommenders构建智能推荐引擎
人工智能·python·tensorflow
聿小翼35 分钟前
selenium-wire 与 googletrans 的爱恨情仇
python
咖啡调调。37 分钟前
模板引擎语法-算术运算
python·django·sqlite
CodeCraft Studio38 分钟前
Excel处理控件Spire.XLS系列教程:Java设置Excel活动工作表或活动单元格
java·python·excel
Doker 多克38 分钟前
Python-Django系列—部件
开发语言·python
Linux运维老纪40 分钟前
Python文件操作及数据库交互(Python File Manipulation and Database Interaction)
linux·服务器·数据库·python·云计算·运维开发