百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

Python 装饰器速览_python装饰器有哪些测试

itomcoil 2025-09-09 00:18 2 浏览

引言

Python 装饰器是一种便捷的工具,用于在不修改函数代码的情况下为其添加额外功能。它在日志记录、性能监控、权限校验等场景中极为有用,能有效减少重复代码,提升代码的可维护性和可读性。

装饰器基础

函数是一等公民

在 Python 中,函数是对象,可以被赋值、传递和返回。例如:

Python

复制

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

say_hello = greet
print(say_hello("Alice"))  # Hello, Alice!

闭包

闭包允许内部函数引用外部函数的变量。例如:

Python

复制

def make_multiplier(n):
    def multiplier(x):
        return x * n
    return multiplier

times2 = make_multiplier(2)
print(times2(5))  # 10

装饰器基本语法

装饰器是一个函数,接收另一个函数作为参数,并返回一个新的函数。使用 @ 语法糖可简化装饰器的使用。

基础装饰器

Python

复制

def simple_decorator(func):
    def wrapper():
        print("Before the function runs")
        func()
        print("After the function runs")
    return wrapper

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

say_hello()

输出:

Text

复制

Before the function runs
Hello!
After the function runs

通用装饰器

对于带参数的函数,装饰器需要使用 *args 和 **kwargs 来接收任意参数:

Python

复制

def universal_decorator(func):
    def wrapper(*args, **kwargs):
        print("Function is running...")
        result = func(*args, **kwargs)
        print("Function finished")
        return result
    return wrapper

@universal_decorator
def add(a, b):
    return a + b

print(add(3, 5))

常见场景与实战示例

日志与调试

Python

复制

def log_decorator(func):
    def wrapper(*args, **kwargs):
        print(f"Calling {func.__name__} with args={args}, kwargs={kwargs}")
        result = func(*args, **kwargs)
        print(f"{func.__name__} returned {result}")
        return result
    return wrapper

@log_decorator
def multiply(a, b):
    return a * b

multiply(3, 4)

性能监控

Python

复制

import time

def timing_decorator(func):
    def wrapper(*args, **kwargs):
        start = time.time()
        result = func(*args, **kwargs)
        end = time.time()
        print(f"{func.__name__} took {end - start:.4f} seconds")
        return result
    return wrapper

@timing_decorator
def slow_function():
    time.sleep(1)
    return "done"

slow_function()

权限校验

Python

复制

def require_admin(func):
    def wrapper(user, *args, **kwargs):
        if not user.get("is_admin"):
            raise PermissionError("Admin access required")
        return func(user, *args, **kwargs)
    return wrapper

@require_admin
def delete_database(user):
    return "Database deleted!"

admin = {"name": "Alice", "is_admin": True}
guest = {"name": "Bob", "is_admin": False}

print(delete_database(admin))   # 正常执行
# print(delete_database(guest))   # 抛出 PermissionError

缓存

Python

复制

def cache_decorator(func):
    cache = {}
    def wrapper(*args):
        if args in cache:
            print("Returning cached result")
            return cache[args]
        result = func(*args)
        cache[args] = result
        return result
    return wrapper

@cache_decorator
def fibonacci(n):
    if n <= 1:
        return n
    return fibonacci(n-1) + fibonacci(n-2)

print(fibonacci(10))  # 第一次计算
print(fibonacci(10))  # 第二次命中缓存

进阶内容

保留函数元信息:functools.wraps

使用 functools.wraps 可以保留原函数的元信息,如 __name__ 和 __doc__:

Python

复制

from functools import wraps

def simple_decorator(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        return func(*args, **kwargs)
    return wrapper

@simple_decorator
def greet(name):
    """Say hello to someone"""
    return f"Hello, {name}!"

print(greet.__name__)  # greet
print(greet.__doc__)   # Say hello to someone

类装饰器

可以用类实现装饰器,通过定义 __call__ 方法让类实例可调用:

Python

复制

class Repeat:
    def __init__(self, times):
        self.times = times

    def __call__(self, func):
        def wrapper(*args, **kwargs):
            result = None
            for _ in range(self.times):
                result = func(*args, **kwargs)
            return result
        return wrapper

@Repeat(3)
def say_hi(name):
    print(f"Hi, {name}!")

say_hi("Alice")

装饰器叠加

多个装饰器的执行顺序是从内到外,即离函数最近的装饰器先执行:

Python

复制

def decorator_a(func):
    def wrapper(*args, **kwargs):
        print("Decorator A before")
        result = func(*args, **kwargs)
        print("Decorator A after")
        return result
    return wrapper

def decorator_b(func):
    def wrapper(*args, **kwargs):
        print("Decorator B before")
        result = func(*args, **kwargs)
        print("Decorator B after")
        return result
    return wrapper

@decorator_a
@decorator_b
def say_hello():
    print("Hello!")

say_hello()

带参数的装饰器工厂

可以通过多层函数实现带参数的装饰器:

Python

复制

from functools import wraps

def repeat(times):
    def decorator(func):
        @wraps(func)
        def wrapper(*args, **kwargs):
            for _ in range(times):
                func(*args, **kwargs)
        return wrapper
    return decorator

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

greet("Alice")

实用库中的装饰器

标准库中的装饰器

  • @staticmethod:定义静态方法。
  • @classmethod:定义类方法。
  • @property:将方法变为只读属性。
  • @functools.lru_cache:缓存函数结果。

第三方框架中的装饰器

  • Flask 的 @app.route:将函数注册为路由处理器。
  • Django 的 @login_required:限制只有登录用户才能访问视图。
  • Click 的命令行装饰器:定义命令和参数。

最佳实践与注意事项

避免滥用

装饰器适用于横切逻辑,避免过度使用,以免降低代码可读性。

始终使用 functools.wraps

使用 functools.wraps 保留原函数的元信息,便于调试和文档生成。

注意副作用

装饰器可能会改变函数的调用方式或引入缓存、权限等逻辑,需注意其副作用。

保持可读性

复杂的装饰器逻辑应清晰注释,必要时可用类装饰器或明确的函数调用替代。

关注性能

装饰器会增加函数调用层级,复杂逻辑或过多装饰器可能影响性能。必要时使用性能分析工具检查。

总结

装饰器是 Python 中一种强大的工具,能有效分离横切逻辑,提高代码的可读性和扩展性。掌握其基础语法、进阶用法及最佳实践,可在实际开发中高效利用装饰器,写出更优雅的代码。

相关推荐

Excel表格,100个常用函数_excel表格各种函数用法

1.SUM:求和函数2.AVERAGE:平均值函数3.MAX:最大值函数4.MIN:最小值函数5.COUNT:计数函数6.IF:条件函数7.VLOOKUP:垂直查找函数8.HLOOKU...

每天学一点Excel2010 (62)—Multinomial、Aggregate、Subtotal

138multinominal助记:英文的“多项式”类别:数学和三角语法:multinominal(number1,[number2],…)参数:1~255个参数number1必需。第1个数值参数...

182.人工智能——构建大模型应用_人工智能:模型与算法

一直认为人工智能的本质其实就是:算法+算力+大数据。算法的尽头是数学,算力是能源、而大数据则是人类共同智慧的而且是有限的宝贵资源,也是决定大模型的能力上限。人工智能不断的发展,也是人类文明进步的必然趋...

Excel伽马函数GAMMA_伽马函数表怎么看

Gamma函数是阶乘函数在实数与复数上扩展的一类函数,通常写作Γ(x)。伽玛函数在分析学、概率论、离散数学、偏微分方程中有重要的作用,属于应用最广泛的函数之一函数公式如下伽玛函数满足递推关系Γ(N+1...

2.黎曼ζ函数与黎曼猜想_黎曼函数的作用
2.黎曼ζ函数与黎曼猜想_黎曼函数的作用

2.黎曼ζ函数与黎曼猜想那么这个让上帝如此吝啬的黎曼猜想究竟是一个什么样的猜想呢?在回答这个问题之前我们先得介绍一个函数:黎曼ζ函数(RiemannZeta-function)。这个函数...

2025-09-09 00:24 itomcoil

嵌入式C语言基础编程—5年程序员给你讲函数,你真的懂函数吗?

本文主要是对C基础编程关于函数的初步讲解,后续会深入讲解C高级相关的概念(C大神可先略过)。本人近期会陆续上传IT编程相关的资料和视频教程,可以关注一下互相交流:CC++Javapython...

进一步理解函数_解读函数

函数的定义和基本调用应该是比较容易理解的,但有很多细节可能令初学者困惑,包括参数传递、返回、函数命名、调用过程等,我们逐个介绍。1.参数传递有两类特殊类型的参数:数组和可变长度的参数。(1)数组数组作...

可以降低阶乘运算复杂度的Stirling公式

转发一个关于Stirling公式的推导方法:Wallis公式是关于圆周率的无穷乘积的公式,但Wallis公式中只有乘除运算,连开方都不需要,形式上十分简单。虽然Wallis公式对π的近似计算没有直接影...

Agent杂谈:Agent的能力上下限及「Agent构建」核心技术栈调研分享~

2025年Agent技术持续演进,已从简单任务处理向具备独立规划、协作能力的智能系统转变。文章从系统设计视角出发,先梳理Agent的核心定义与架构框架,再深入分析决定其能力上下限的关键因素...

无炮塔的“S”坦克/Strv-103主战坦克

  20世纪50年代,瑞典陆军为了对付当时苏联T-54坦克,着手研制了一种无炮塔坦克——“S”坦克(瑞典编号为Strv103),并于1967年正式投产。这种坦克具有创新的设计思想,打破了传统的设计方...

shell——字符串操作_shell字符串处理命令

str="abc123abcABC"#计算字符串的长度echo${#str}#12exprlength$strexpr"$str":".*&#...

XSS的两种攻击方式及五种防御方式

跨站脚本攻击指的是自己的网站运行了外部输入代码攻击原理是原本需要接受数据但是一段脚本放置在了数据中:该攻击方式能做什么?获取页面数据获取Cookies劫持前端逻辑发送请求到攻击者自己的网站实现资料的盗...

C语言字符数组和字符串_c语言中的字符数组

用来存放字符的数组称为字符数组,例如:charc[10];字符数组也可以是二维或多维数组。例如:charc[5][10];字符数组也允许在定义时进行初始化,例如:charc[10]={'c',...

Python 和 JS 有什么相似?_python跟js

Python是一门运用很广泛的语言,自动化脚本、爬虫,甚至在深度学习领域也都有Python的身影。作为一名前端开发者,也了解ES6中的很多特性借鉴自Python(比如默认参数、解构赋值、...

【python】装饰器的原理_python装饰器详细教程

装饰器的原理是利用了Python的函数特性,即函数可以作为参数传递给另一个函数,也可以作为另一个函数的返回值。装饰器本质上是一个接受一个函数作为参数,并返回一个新函数的函数。这个新函数通常会在执行原函...