当前位置:首页 > Python > 正文内容

Python 装饰器高级应用与实战技巧

admin3周前 (03-24)Python22

装饰器是 Python 中最优雅的语法特性之一,它允许我们在不修改原始函数代码的情况下,为函数添加额外的功能。本文将从实际应用场景出发,深入探讨装饰器的高级用法。

基础回顾

装饰器的本质是一个接受函数作为参数,并返回一个新函数的高阶函数。最简单的装饰器可以这样写:

def timer(func):
    import time
    
    def wrapper(*args, **kwargs):
        start = time.time()
        result = func(*args, **kwargs)
        end = time.time()
        print(f"{func.__name__} 执行时间: {end - start:.4f}秒")
        return result
    
    return wrapper

@timer
def slow_function():
    import time
    time.sleep(0.5)
    return "完成"

带参数的装饰器

当装饰器需要接受自定义参数时,我们需要再包裹一层函数:

def retry(max_attempts=3, delay=1):
    def decorator(func):
        import time
        
        def wrapper(*args, **kwargs):
            attempts = 0
            while attempts < max_attempts:
                try:
                    return func(*args, **kwargs)
                except Exception as e:
                    attempts += 1
                    if attempts == max_attempts:
                        raise
                    print(f"第 {attempts} 次失败,{delay} 秒后重试...")
                    time.sleep(delay)
        
        return wrapper
    return decorator

@retry(max_attempts=2, delay=0.5)
def unstable_api():
    import random
    if random.random() > 0.7:
        raise Exception("API 暂时不可用")
    return "API 响应成功"

类装饰器

使用类作为装饰器可以更好地维护状态:

class CacheDecorator:
    def __init__(self, max_size=100):
        self.cache = {}
        self.max_size = max_size
        self.access_order = []
    
    def __call__(self, func):
        def wrapper(*args, **kwargs):
            # 创建缓存键
            key = (args, frozenset(kwargs.items()))
            
            if key in self.cache:
                return self.cache[key]
            
            result = func(*args, **kwargs)
            
            # 更新缓存
            if len(self.cache) >= self.max_size:
                oldest = self.access_order.pop(0)
                del self.cache[oldest]
            
            self.cache[key] = result
            self.access_order.append(key)
            
            return result
        
        return wrapper

@CacheDecorator(max_size=50)
def expensive_calculation(n):
    print(f"计算 {n} 的斐波那契数...")
    if n <= 1:
        return n
    return expensive_calculation(n-1) + expensive_calculation(n-2)

装饰器堆叠

多个装饰器可以堆叠使用,执行顺序从内到外:

def log_call(func):
    def wrapper(*args, **kwargs):
        print(f"调用 {func.__name__}({args})")
        return func(*args, **kwargs)
    return wrapper

def validate_types(*type_args):
    def decorator(func):
        def wrapper(*args, **kwargs):
            for i, (arg, expected_type) in enumerate(zip(args, type_args)):
                if not isinstance(arg, expected_type):
                    raise TypeError(f"参数 {i} 应该是 {expected_type}")
            return func(*args, **kwargs)
        return wrapper
    return decorator

@log_call
@validate_types(int, int)
def add_numbers(a, b):
    return a + b

保留原函数元数据

使用 functools.wraps 可以保留原函数的元信息:

import functools

def admin_required(func):
    @functools.wraps(func)
    def wrapper(*args, **kwargs):
        # 检查权限逻辑
        print("检查管理员权限...")
        return func(*args, **kwargs)
    return wrapper

@admin_required
def delete_user(user_id):
    """删除指定用户"""
    return f"用户 {user_id} 已删除"

# 现在 delete_user.__name__ 和 __doc__ 都能正确获取

实战应用:权限控制装饰器

from functools import wraps

class PermissionDenied(Exception):
    pass

def require_permission(permissions):
    """权限检查装饰器"""
    def decorator(func):
        @wraps(func)
        def wrapper(*args, **kwargs):
            user = kwargs.get('user')
            if not user or not any(p in user.get('permissions', []) for p in permissions):
                raise PermissionDenied(f"需要权限: {permissions}")
            return func(*args, **kwargs)
        return wrapper
    return decorator

# 使用示例
class UserService:
    @require_permission(['user:read'])
    def get_user(self, user_id, user):
        return {"id": user_id, "name": "测试用户"}
    
    @require_permission(['user:write'])
    def update_user(self, user_id, data, user):
        return {"id": user_id, **data}

实战应用:性能监控装饰器

import time
import functools
from collections import defaultdict

class PerformanceMonitor:
    def __init__(self):
        self.stats = defaultdict(list)
    
    def track(self, func):
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
            start = time.perf_counter()
            result = func(*args, **kwargs)
            end = time.perf_counter()
            duration = end - start
            self.stats[func.__name__].append(duration)
            return result
        return wrapper
    
    def get_stats(self):
        report = {}
        for func_name, times in self.stats.items():
            report[func_name] = {
                'total_calls': len(times),
                'avg_time': sum(times) / len(times),
                'max_time': max(times),
                'min_time': min(times)
            }
        return report

# 使用示例
monitor = PerformanceMonitor()

@monitor.track
def process_data(n):
    import random
    time.sleep(random.uniform(0.01, 0.1))
    return n * 2

总结

装饰器是 Python 中强大的代码复用工具。通过合理使用装饰器,我们可以:

1. 保持代码简洁和可维护性

2. 实现横切关注点(日志、缓存、权限等)

3. 提高代码的可读性和复用性

在实际项目中,建议将常用装饰器整理到单独的模块中,并配合 functools.wraps 使用,以保持良好的代码质量。

相关文章

Python 上下文管理器:从入门到实战

在 Python 编程中,资源管理是一个永恒的话题。无论是打开文件、连接数据库,还是获取网络资源,我们都需要确保在使用完毕后正确释放这些资源。传统的 try-finally 模式虽然有效,但代码冗长且...

Python 上下文管理器的 5 个实用技巧,让你的代码更优雅

在 Python 编程中,上下文管理器(Context Manager)是一个优雅的资源管理工具。你可能已经熟悉最常见的用法——使用 with 语句打开文件,但上下文管理器的能力远不止于此。今天,我将...

Python 装饰器的高级应用与实战技巧

装饰器本质上是接受函数作为参数并返回新函数的高阶函数。理解这一点是掌握装饰器的关键。让我们从基础开始,逐步深入到高级应用。首先,我们需要理解函数在 Python 中是一等公民。这意味着函数可以像其他对...

Python 装饰器完全指南:从入门到实战

装饰器本质上是一个接受函数作为参数并返回新函数的高阶函数。理解这一点是掌握装饰器的关键。想象一下,你有一个函数,你想在它执行前后添加一些额外的逻辑,比如日志记录、性能测试、权限验证等。装饰器就是为这种...

Python 生成器进阶:理解 yield 与构建高效迭代器

在 Python 开发中,我们经常需要处理大量数据或流式数据,如果一次性将所有数据加载到内存中,不仅会占用大量内存空间,还可能导致程序运行缓慢甚至崩溃。生成器(Generator)正是解决这个问题的利...

Python 数据处理实战:从零开始掌握 Pandas 核心操作

在现代数据驱动的世界中,处理和分析结构化数据已成为必备技能。无论你是数据分析师、机器学习工程师还是科研人员,Pandas 都是你工具箱中不可或缺的利器。与 Excel 相比,Pandas 能够轻松处理...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。