Python中的装饰器是什么?

本文详细介绍了Python中的装饰器,包括其工作原理、基本用法、参数传递、多装饰器应用以及一个计算函数执行时间的实际例子。装饰器是一种强大工具,用于在不修改原函数代码的情况下扩展其功能。
部署运行你感兴趣的模型镜像

引言

在Python的世界里,装饰器是一种非常强大而优雅的工具,它允许程序员在不修改原有函数代码的前提下,增加函数的功能。

本文将向大家详细介绍什么是装饰器,它是如何工作的,以及如何在自己的代码中使用装饰器。

什么是装饰器

装饰器本质上是一个函数,它可以让其他函数在不改变自身代码的情况下增加额外的功能。装饰器的使用非常广泛,比如:记录日志、性能测试、事务处理、缓存、权限校验等等。

装饰器的基本使用

装饰器的使用非常简单,只需要在需要被装饰的函数上方添加一个@装饰器名即可。下面通过一个简单的例子来说明装饰器的基本使用。

def my_decorator(func):
    def wrapper():
        print("Something is happening before the function is called.")
        func()
        print("Something is happening after the function is called.")
    return wrapper

@my_decorator
def say_hello():
    print("Hello!")

say_hello()

运行这段代码,输出将会是:

Something is happening before the function is called.
Hello!
Something is happening after the function is called.

在这个例子中,my_decorator就是一个装饰器,它接受一个函数func作为参数,并返回一个新的函数wrapperwrapper函数中加入了在func执行前后需要执行的代码。通过@my_decorator,我们将这个装饰器应用到了say_hello函数上。

装饰器中传递参数

如果我们的函数有参数怎么办?装饰器也能很好地处理这种情况。看下面的例子:

def my_decorator(func):
    def wrapper(*args, **kwargs):
        print("Something is happening before the function is called.")
        result = func(*args, **kwargs)
        print("Something is happening after the function is called.")
        return result
    return wrapper

@my_decorator
def say(message):
    print(message)

say("Hello with argument!")

这段代码中,wrapper函数使用了*args**kwargs来接受任意数量和类型的参数,这使得装饰器变得更加灵活。

使用多个装饰器

Python也支持给一个函数同时使用多个装饰器,如下所示:

def decorator_one(func):
    def wrapper():
        print("Decorator one!")
        func()
    return wrapper

def decorator_two(func):
    def wrapper():
        print("Decorator two!")
        func()
    return wrapper

@decorator_one
@decorator_two
def say_hello():
    print("Hello!")

say_hello()

当使用多个装饰器时,它们的执行顺序是从近到远,即先decorator_twodecorator_one

真实例子:计算函数执行所需时间
现在通过一个更实际的例子来进一步理解装饰器:假设我们要编写一个装饰器,用于计算任何函数执行所需的时间。这在性能测试时非常有用。

import time

# 创建一个装饰器,它会计算并打印函数执行所需的时间
def time_decorator(func):
    def wrapper(*args, **kwargs):
        start_time = time.time()  # 记录函数开始执行的时间
        result = func(*args, **kwargs)  # 执行函数
        end_time = time.time()  # 记录函数执行完成的时间
        print(f"{func.__name__} took {end_time - start_time} seconds to run.")
        return result
    return wrapper

# 使用装饰器
@time_decorator
def long_running_function(number):
    sum = 0
    for i in range(number):
        sum += i
    return sum

# 调用函数
print(long_running_function(1000000))

在这个例子中,我们首先定义了一个名为time_decorator的装饰器。这个装饰器接受一个函数func作为参数,并返回一个新的函数wrapperwrapper函数会记录被装饰的函数执行前后的时间,从而计算出函数执行所需的总时间,并打印出来。

然后,我们创建了一个long_running_function函数,它接受一个参数并执行一个简单的累加操作。通过在这个函数前面添加@time_decorator,我们就把装饰器应用到了long_running_function上。

当我们调用long_running_function(1000000)时,装饰器会自动计算并打印出这个函数执行所需的时间。这样,我们就可以很容易地监控任何函数的性能了。

总结

装饰器是Python中一种非常强大的工具,它能够帮助我们以非常优雅的方式增强函数的功能,而不需要修改函数本身的代码。

学习资源推荐
除了上述分享,学好 Python 不论是就业还是做副业赚钱都不错,但要学会 Python 还是要有一个学习规划。最后大家分享一份全套的 Python 学习资料,给那些想学习 Python 的小伙伴们一点帮助!

包括:Python激活码+安装包、Python web开发,Python爬虫,Python数据分析,人工智能、自动化办公等学习教程。带你从零基础系统性的学好Python!

👉Python所有方向的学习路线👈

Python所有方向路线就是把Python常用的技术点做整理,形成各个领域的知识点汇总,它的用处就在于,你可以按照上面的知识点去找对应的学习资源,保证自己学得较为全面。(全套教程文末领取)

在这里插入图片描述
👉Python学习视频600合集👈

观看零基础学习视频,看视频学习是最快捷也是最有效果的方式,跟着视频中老师的思路,从基础到深入,还是很容易入门的。

在这里插入图片描述

温馨提示:篇幅有限,已打包文件夹,获取方式在:文末

👉Python70个实战练手案例&源码👈

光学理论是没用的,要学会跟着一起敲,要动手实操,才能将自己的所学运用到实际当中去,这时候可以搞点实战案例来学习。

在这里插入图片描述

👉Python大厂面试资料👈

我们学习Python必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有阿里大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。

在这里插入图片描述

在这里插入图片描述

👉Python副业兼职路线&方法👈

学好 Python 不论是就业还是做副业赚钱都不错,但要学会兼职接单还是要有一个学习规划。

在这里插入图片描述

👉 这份完整版的Python全套学习资料已经上传,朋友们如果需要可以V扫描下方二维码联系领取
保证100%免费

您可能感兴趣的与本文相关的镜像

Python3.8

Python3.8

Conda
Python

Python 是一种高级、解释型、通用的编程语言,以其简洁易读的语法而闻名,适用于广泛的应用,包括Web开发、数据分析、人工智能和自动化脚本

### 装饰器的概念 装饰器(Decorator)是 Python 中一种特殊的函数或类,用于在不修改原有函数或类代码的前提下,为其添加额外的功能。装饰器本质上是一个闭包函数,它接收一个函数作为参数,并返回一个新的函数。这种机制使得开发者可以在函数调用前后执行一些额外操作,例如日志记录、性能测试、权限控制等。 装饰器的核心思想是通过函数包装的方式扩展函数行为,而不是修改函数本身。这一特性使得代码更加简洁、可维护,并且符合开放封闭原则[^3]。 ### 装饰器的基本语法 装饰器的语法使用 `@` 符号,紧接在函数定义之前。例如: ```python @decorator_func def my_func(): pass ``` 上述代码等价于: ```python def my_func(): pass my_func = decorator_func(my_func) ``` 这意味着装饰器 `decorator_func` 会接收 `my_func` 函数作为参数,并将其包装成一个新的函数对象[^3]。 ### 无参装饰器的使用 最简单的装饰器不接受任何参数,仅用于包装函数。例如,定义一个简单的装饰器来记录函数执行时间: ```python import time def timer(func): def wrapper(*args, **kwargs): start_time = time.time() result = func(*args, **kwargs) end_time = time.time() print(f"函数 {func.__name__} 执行耗时: {end_time - start_time:.4f} 秒") return result return wrapper @timer def say_hello(): print("Hello World") say_hello() ``` 输出结果为: ``` Hello World 函数 say_hello 执行耗时: 0.0001 秒 ``` 在这个例子中,`timer` 是一个无参装饰器,它将 `say_hello` 函数包装成一个新的函数 `wrapper`,并在调用前后增加了计时逻辑[^4]。 ### 带参装饰器的使用 有时装饰器本身也需要接受参数,这就需要使用“装饰器工厂”来生成装饰器装饰器工厂是一个返回装饰器函数的函数。例如,定义一个带参数的装饰器,根据不同的日志级别输出信息: ```python def logger(level): def decorator(func): def wrapper(*args, **kwargs): if level == "INFO": print(f"[INFO] 正在调用函数 {func.__name__}") elif level == "DEBUG": print(f"[DEBUG] 正在调用函数 {func.__name__}") return func(*args, **kwargs) return wrapper return decorator @logger("INFO") def greet(name): print(f"Hello, {name}") greet("Alice") ``` 输出结果为: ``` [INFO] 正在调用函数 greet Hello, Alice ``` 该示例中,`logger("INFO")` 是一个装饰器工厂,它返回一个装饰器函数,再由该装饰器函数包装 `greet` 函数[^1]。 ### 类作为装饰器 除了使用函数实现装饰器,还可以使用类来实现装饰器。类装饰器通常需要实现 `__call__` 方法。例如: ```python class MyClassDecorator: def __init__(self, func): self.func = func def __call__(self, *args, **kwargs): print("调用函数前执行") result = self.func(*args, **kwargs) print("调用函数后执行") return result @MyClassDecorator def say_hi(): print("Hi") say_hi() ``` 输出结果为: ``` 调用函数前执行 Hi 调用函数后执行 ``` 该装饰器通过类的 `__call__` 方法实现了函数包装功能[^2]。 ### 装饰器的应用场景 装饰器广泛应用于以下场景: - **日志记录**:在函数调用前后记录日志信息。 - **权限控制**:在执行函数前检查用户权限。 - **缓存机制**:缓存函数执行结果以提高性能。 - **性能测试**:统计函数执行时间。 - **路由注册**:在 Web 框架中注册路由。 装饰器提供了一种优雅且灵活的方式来增强函数或类的功能,是 Python 编程中非常重要的特性之一。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值