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

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

admin2个月前 (03-24)Python63

装饰器是 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 教程] OpenCV 实战:图像与视频文件处理

OpenCV 实战:图像与视频文件处理本文详细介绍如何使用 OpenCV 处理图像和视频文件,包括读取、显示、保存等操作。一、图像文件操作1.1 读取图像import cv2 #&nb...

[Python 教程] OpenCV 绘图教程:图形与文本标注

OpenCV 绘图教程:图形与文本标注本文介绍如何在 OpenCV 中绘制各种图形和添加文本,用于图像标注和可视化。一、绘制基本图形1.1 创建画布import cv2 import&nb...

Python 高级技巧:让你的代码更优雅高效

# Python 高级技巧:让你的代码更优雅高效 在 Python 编程的世界里,掌握基础语法只是第一步。真正的高手懂得运用高级技巧,让代码更简洁、更高效、更易维护。今天,我将分享一些实用且不那么广为...

Python异步编程入门与实战

异步编程是一种并发执行的编程模式,它允许程序在等待耗时操作(如网络请求、文件读写)时,继续执行其他任务。Python 3.5引入了async/await语法,使得异步编程变得更加直观和易于理解。为什么...

Python 装饰器:从原理到高级实战完全指南

Python 装饰器是一种强大的语法糖,它可以在不修改原函数代码的情况下,为函数添加额外的功能。装饰器的本质是一个接受函数作为参数,并返回一个新函数的高阶函数。 装饰器的基本原理 装饰器的工作原理...

Python dataclass:现代面向对象编程的最佳实践

在传统的 Python 面向对象编程中,当我们需要创建一个主要用于存储数据的类时,往往需要编写大量的样板代码。我们需要手动定义 `__init__` 方法来初始化属性,实现 `__repr__` 方...

发表评论

访客

看不清,换一张

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