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

大语言模型解释Python的 类装饰器

itomcoil 2025-09-06 02:14 4 浏览

一、什么是类装饰器?

在 Python 中,装饰器(Decorator)是一种高阶函数,它接受另一个对象(通常是函数或类),并返回一个经“增强”处理后的新对象。我们常见的是对函数进行装饰:

@my_decorator
def foo(...):
...

同样的机制也可以应用于类定义,只需将装饰器放置在类定义前即可:

@my_class_decorator
class MyClass:
...

当 Python 解释器读取这段代码时,首先会执行 my_class_decorator(MyClass),然后用其返回值(通常是一个修改过的新类对象)替代原类定义。这个机制让开发者可以在类创建阶段插入自定义行为,提升代码复用性和元编程能力。

二、类装饰器的典型应用场景

类装饰器的主要应用场景包括但不限于:

  • 自动添加属性或方法
  • 统一初始化逻辑(如日志、性能监控、计数)
  • 实现动态元编程(例如自动混入 Mixin、序列化支持)

下面通过三个实用示例详细讲解类装饰器的设计与使用。

三、实用示例解析

示例 1:为每个实例自动记录创建时间

我们可以利用类装饰器来为所有实例统一加上时间戳功能,使其在每次实例化时自动记录创建时间。

装饰器实现

import datetime
def timestamp_decorator(cls):
"""
给类添加一个 `created_at` 属性,在 __init__ 中记录对象创建时间。
"""
original_init = cls.__init__
def new_init(self, *args, **kwargs):
# 先调用原始 __init__ 方法完成原本的初始化过程
original_init(self, *args, **kwargs)
# 然后添加当前时间戳
self.created_at = datetime.datetime.now()
# 替换类中的 __init__
cls.__init__ = new_init
return cls

使用方式

@timestamp_decorator
class User:
def __init__(self, name):
self.name = name
u = User('Alice')
print(u.created_at) # 输出类似:2025-04-05 10:30:45.123456

解释说明

该装饰器会保存原有 __init__ 方法,并在其基础上新增设置时间戳的功能。这种做法无需侵入原有类结构,便于功能扩展和后期维护。

示例 2:给类的所有公开方法添加日志打印

有时我们希望在每次调用某个类的方法时自动输出日志信息,便于调试或审计。

装饰器实现

import functools
def log_method_decorator(cls):
"""
为类中所有公开方法(非下划线开头)包装一层日志输出。
"""
for name, method in cls.__dict__.items():
if callable(method) and not name.startswith('_'):
@functools.wraps(method)
def wrapper(self, *args, **kwargs):
print(f"[LOG] 调用 {cls.__name__}.{name},参数: {args},关键字参数: {kwargs}")
return method(self, *args, **kwargs)
setattr(cls, name, wrapper)
return cls

使用方式

@log_method_decorator
class Calculator:
def add(self, a, b):
return a + b
def mul(self, a, b):
return a * b
calc = Calculator()
calc.add(3, 4) # 输出日志并返回 7
calc.mul(2, 5) # 输出日志并返回 10

解释说明

通过 setattr 动态地替换了每个公开方法,同时使用 functools.wraps 确保封装后的函数保留了原有签名和文档说明。这一技巧特别适合构建通用的日志系统或者拦截调用链。

示例 3:动态为类添加 JSON 序列化能力(Mixin 实现)

有时候我们要给多个类批量添加相同的功能,比如使类支持以 JSON 格式输出数据。

装饰器实现

import json
def serializable_decorator(cls):
"""
为类动态添加 to_json() 方法,用于将属性转为 JSON 字符串。
"""
def to_json(self):
# 只导出不含下划线前缀的属性
public_attrs = {k: v for k, v in self.__dict__.items() if not k.startswith('_')}
return json.dumps(public_attrs, ensure_ascii=False)
# 将方法绑定到类上
cls.to_json = to_json
return cls

使用方式

@serializable_decorator
class Person:
def __init__(self, name, age):
self.name = name
self.age = age
p = Person("Bob", 30)
print(p.to_json()) # 输出 {"name": "Bob", "age": 30}

解释说明

此方式模拟了一种常见的 Mixin 模式:通过类装饰器动态注入通用功能。相比于继承多重 Mixin,类装饰器可以更灵活、优雅地控制附加功能的应用范围。

四、类装饰器的总结

特性

描述

语法结构

用 @decorator 放在类声明之前

工作原理

是一个接收类作为参数并返回新类的函数

常见用途

自动注入方法/属性、统一处理初始化、增强行为、元编程

优势

低耦合、易复用、不破坏类原有结构、适合横切关注点

类装饰器允许开发者在类创建期间就施加一些变更,是 Python 面向切面编程(AOP)的一种优雅实践方式。

五、拓展建议

若希望进一步了解装饰器机制,可研究以下高级主题:

  1. 带参数的类装饰器(Decorator Factory)
  • @add_attributes(version='1.0', author='dev')
    class MyService:
    ...
  1. 使用类作为装饰器(Callable 类)
  • class MyDecorator:
    def __call__(self, cls):
    ...
  1. 组合多种装饰器
  • @log_call
    @auto_timestamp
    @serialize_support
    class Service:
    ...

通过这些进阶方式,你可以构建出强大的、高度可配置的类增强逻辑。

相关推荐

Filter函数在WPS里的正确用法,官方教程里都没有说......

Filter函数是office365新增的筛选函数,WPS也紧跟添加了它。但在二个软件中的使用方法却完全不同。office365有单元格溢出功能,只需要输入一个Filter公式即可完成数据筛选。但在W...

跳过VLOOKUP天坑!FILTER函数10个招式让同事以为你开了外挂?

还在为VLOOKUP的"一对多"限制头疼?是否还在为INDEX+MATCH的嵌套抓狂?今天教你用Excel新晋顶流——FILTER函数,10个高能用法让你秒变数据操控大师!用法1:精准...

Filter函数的三种用法,比用VLOOKUP一对多查询,更加灵活方便

文章最后有彩蛋!好礼相送!Excel秘籍大全,正文开始FILTER函数可以基于定义的条件筛选一系列数据。在没有filter函数之前,如果实现一对多查询,常见的是构建辅助列,然后使用VLOOKUP+R...

Filter函数公式,快速实现订单核对,1分钟学会

举个例子,我们有一份公司所有的订单源数据表格,这里我们只用两列信息来模拟,实际可能有很多列数据,几百行数据然后我们有另外一个表,里面有部分已经处理过的订单数据,如下所示,这里举例是4个,实际可能有上百...

FILTER函数结合及经典用法2:一对多筛选

FILTER经典用法2:一对多筛选。FILTER函数的经典用法2:一对多的筛选。比如左边这个表格,需要根据部门筛选出每个部门的人员,应该怎样做?·把鼠标放在单元格内,在编辑栏输入等于FILTER。·第...

干掉VLOOKUP,FILTER函数9大用法全解析!

1.单条件基础筛选场景:筛选销量>5000的记录公式:=VSTACK(A1:D1,FILTER(A2:D9,D2:D9>5500))解析:A2:D9为需要筛选的数据区域,D2:D9&...

Excel新函数公式Filter,秒杀VLOOKUP,人人必学

以前VLOOKUP公式是必学的公式,自从新版本更新之后,VLOOKUP已经变得可有可无了,但是新出来的Filter函数公式,你必须学会,它非常的强大,工作中用到非常频繁1、Filter公式背景在学会这...

第一讲:filter的基本用法及拓展_filter详解

全能查找函数filter的基本用法及拓展初学者,务必观看。进阶者,可互相学习,欢迎在回复中补充新用法。首次撰写此函数相关内容,若有不足之处,请予以指教,请勿诋毁,多谢。提示:以下内容以WPS最新版本为...

测一测你是什么粒子?_测测你是什么质

大亚湾实验。|图片来源:RoyKaltschmidt,LawrenceBerkeleyNationalLaboratory/WikimediaCommons2020年12月12日,大亚湾...

SpringBoot如何处理配置文件的密文

在SpringBoot应用中,直接在配置文件(如application.yml或application.properties)中明文存储数据库密码、API密钥等敏感信息是严重的安全风险,...

大语言模型解释Python的 类装饰器

一、什么是类装饰器?在Python中,装饰器(Decorator)是一种高阶函数,它接受另一个对象(通常是函数或类),并返回一个经“增强”处理后的新对象。我们常见的是对函数进行装饰:@my_dec...

Thymeleaf_thymeleaf属于前端吗

一、Thymeleaf简介Thymeleaf是用来开发Web和独立环境项目的服务器端的Java模版引擎Spring官方支持的服务的渲染模板中,并不包含jsp。而是Thymeleaf和Freemarke...

Win9去哪了?Win10避讳Windows95、98

10月1日,微软在旧金山发布了新一代操作系统预览版。但不是名为Windows9,而是win10,有业内人士猜测,跳过9而取10为命名是为了预示十全十美。可是小编还觉得9还代表长长久久呢!恐怕这里又说...

仓颉编程练习-字符串操作_仓颉编译器

main.cj:importstd.convert.Parsablemain():Int64{//字符串比较lets1:String="abc"...

一课译词:断断续续_一课译词:断断续续的意思

PhotobyMikefromPexels“断断续续”,或“时断时续”,意思是时而中断,时而继续地接连下去(continuefromtimetotime)。与英文惯用语“fitsan...