大语言模型解释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)的一种优雅实践方式。
五、拓展建议
若希望进一步了解装饰器机制,可研究以下高级主题:
- 带参数的类装饰器(Decorator Factory):
- @add_attributes(version='1.0', author='dev')
class MyService:
...
- 使用类作为装饰器(Callable 类):
- class MyDecorator:
def __call__(self, cls):
...
- 组合多种装饰器:
- @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...
- 一周热门
- 最近发表
- 标签列表
-
- ps图案在哪里 (33)
- super().__init__ (33)
- python 获取日期 (34)
- 0xa (36)
- super().__init__()详解 (33)
- python安装包在哪里找 (33)
- linux查看python版本信息 (35)
- python怎么改成中文 (35)
- php文件怎么在浏览器运行 (33)
- eval在python中的意思 (33)
- python安装opencv库 (35)
- python div (34)
- sticky css (33)
- python中random.randint()函数 (34)
- python去掉字符串中的指定字符 (33)
- python入门经典100题 (34)
- anaconda安装路径 (34)
- yield和return的区别 (33)
- 1到10的阶乘之和是多少 (35)
- python安装sklearn库 (33)
- dom和bom区别 (33)
- js 替换指定位置的字符 (33)
- python判断元素是否存在 (33)
- sorted key (33)
- shutil.copy() (33)