VB.net 2010 视频教程 VB.net 2010 视频教程 python基础视频教程
SQL Server 2008 视频教程 c#入门经典教程 Visual Basic从门到精通视频教程
当前位置:
首页 > Python基础教程 >
  • Python之装饰器

装饰器
今天我们来学习以下Python里面的高阶函数,装饰器decorator;它是一种强大的工具,可以修改、扩展函数(方法)的行为,并且无需修改原函数(方法)的代码;身为高阶函数,必然有高级的地方,即函数可以作为参数传递,也可以当作返回值返回

别看上面写的一堆话,它的作用其实很简单

代码复用,通用的功能抽象为装饰器
代码解耦,核心逻辑和附加功能分离开
动态扩展,运行时候修改函数的行为
下面就让我们进行实战以下,看看装饰器到底是个怎么回事

定义、使用装饰器

# 定义装饰器
def decorator(func):
    def wrapper(*args, **kwargs):
        # 在调用原始函数之前执行的操作
        print("Before calling the function")
        result = func(*args, **kwargs)
        # 在调用原始函数之后执行的操作
        print("After calling the function")
        return result
    return wrapper

# 使用装饰器
@decorator
def say_hello(name):
    return f"Hello, {name} "

print(say_hello("Michael"))


"""
Before calling the function
After calling the function
Hello, Michael 
"""

手动应用装饰器

# 定义装饰器
def decorator(func):
    def wrapper(*args, **kwargs):
        # 在调用原始函数之前执行的操作
        print("Before calling the function")
        result = func(*args, **kwargs)
        # 在调用原始函数之后执行的操作
        print("After calling the function")
        return result
    return wrapper


def say_hello(name):
    return f"Hello, {name} "
    

# 手动应用装饰器
decorated_function = decorator(say_hello)
print(decorated_function("Anna"))

"""
Before calling the function
After calling the function
Hello, Michael 
"""

带参数的装饰器

# 带参数的装饰器
# 外层函数接受装饰器参数
# 内层函数接受目标函数
def repart(num_times):
    def decorator(func):
        def wrapper(*args, **kwargs):
            for _ in range(num_times):
                result = func(*args, **kwargs)
            return result

        return wrapper

    return decorator


@repart(3)
def greet(name):
    print(f"Hello, {name}")


greet("Anna")

"""
Hello, Anna
Hello, Anna
Hello, Anna
"""

类装饰器

# 类装饰器
# 通过实现__call__方法来装饰函数

class Mydecorator:
    def __init__(self, func):
        self.func = func

    def __call__(self, *args, **kwargs):
        print("Before calling the function")
        result = self.func(*args, **kwargs)
        print("After calling the function")
        return result


@Mydecorator
def say_hello(name):
    return f"Hello {name}"


print(say_hello("John"))


"""
Before calling the function
After calling the function
Hello John
"""

保留函数的元信息
但是使用装饰器后,原始函数的元信息会被覆盖(__name__和__doc__等)

解决办法就是下面的functools.wraps

# 保留函数的元信息
from functools import wraps


def my_decorator(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        print("Before calling the function")
        result = func(*args, **kwargs)
        print("After calling the function")
        return result

    return wrapper


@my_decorator
def say_hello(name):
    """Greet someone by name."""
    print(f"Hello, {name}!")


print(say_hello.__name__) # say_hello
print(say_hello.__doc__) # Greet someone by name.

当然了,有的同学会问,我可不可以使用两个、三个装饰器呢;答案是当然可以的;我们来看一下DeepSeek给出的例子

# 装饰器 1:日志记录
def log_decorator(func):
    def wrapper(*args, **kwargs):
        # 记录函数调用
        print(f"Log: Calling function {func.__name__} with args: {args}, kwargs: {kwargs}")
        # 调用原始函数
        result = func(*args, **kwargs)
        # 记录函数返回结果
        print(f"Log: Function {func.__name__} returned: {result}")
        return result
    return wrapper


# 装饰器 2:权限验证
def auth_decorator(func):
    def wrapper(*args, **kwargs):
        # 模拟权限检查
        user = "admin"  # 假设当前用户是 admin
        if user == "admin":
            print("Auth: User is authorized.")
            # 调用原始函数
            result = func(*args, **kwargs)
            return result
        else:
            raise PermissionError("Auth: Unauthorized access.")
    return wrapper


# 装饰器 3:性能测试
def timing_decorator(func):
    import time

    def wrapper(*args, **kwargs):
        # 记录开始时间
        start_time = time.time()
        # 调用原始函数
        result = func(*args, **kwargs)
        # 记录结束时间
        end_time = time.time()
        # 计算并输出执行时间
        print(f"Timing: Function {func.__name__} took {end_time - start_time:.4f} seconds")
        return result
    return wrapper


# 应用多个装饰器
@log_decorator
@auth_decorator
@timing_decorator
def greet(name):
    # 模拟一个耗时操作
    import time
    time.sleep(1)
    return f"Hello, {name}!"

# 等价于
greet = log_decorator(auth_decorator(timing_decorator(greet)))

# 测试代码
if __name__ == "__main__":
    try:
        print(greet("Alice"))
    except PermissionError as e:
        print(e)

我们可以看下面的输出信息,首先是打印Log、其次是权限认证、时间;最后就是Log结束;没错通过输出信息我们可以看出来,多个装饰器的时候,执行顺序是从上到下的;返回的时候是从下到上

当调用 greet("Alice") 时,装饰器会按照以下顺序执行:
log_decorator 的 wrapper。
auth_decorator 的 wrapper。
timing_decorator 的 wrapper。
原始函数 greet。
返回时,顺序相反:
timing_decorator 的 wrapper。
auth_decorator 的 wrapper。
log_decorator 的 wrapper。
下面我们来看一下多装饰器的执行流程

调用greet("Alice")
进入log_decorator的wapper,记录函数的调用
进入auth_decorator的wapper,检查权限
进入timing_decorator的wapper,记录开始时间
最后调用原始函数greet
返回结果流程
原始函数greet返回结果
timing_decorator的wapper记录结束时间并输出执行时间
auth_decorator的wapper返回结果
log_decorator的wapper记录返回结果并输出

Log: Calling function wrapper with args: ('Alice',), kwargs: {}
Auth: User is authorized.
Timing: Function greet took 1.0002 seconds
Log: Function wrapper returned: Hello, Alice!
Hello, Alice!

装饰器就到这里了,这里只是简单的例子,具体还需要私底下找一些实例来进行操作,才能更好的记忆深刻

来源:https://www.cnblogs.com/maoning/p/18785474


相关教程