当前位置: 首页 > news >正文

python之dataclasses

一、场景

dataclasses模块提供了一种方便的方法来创建和管理数据对象
它可以帮助开发者更容易地创建简单的类,同时提供了一些实用的功能,例如自动实现__init__()、repr()、eq()等方法。

  • 数据容器:如果您需要一个简单的类来存储一些数据,例如配置信息、用户信息、数据记录等,那么使用dataclass是非常合适的。dataclass可以自动为您生成适当的属性和方法,以便您可以轻松地访问和修改数据。

  • 数据对象:如果您的代码需要处理大量的数据对象,例如在数据分析、机器学习、自然语言处理等领域中,那么使用dataclass可以使代码更加清晰易读,减少手动编写大量的属性和方法的重复性工作。

  • 序列化和反序列化:dataclass可以帮助您更轻松地将对象序列化为JSON、XML等格式,并在需要时将其反序列化。这使得处理不同系统之间的数据交换变得更加容易。

  • 数据验证和清理:在某些情况下,您可能需要在将数据存储到数据库或发送到其他系统之前对其进行验证和清理。使用dataclass可以使这个过程更加简单和可靠。

二、基础用法

from dataclasses import dataclass, field, asdict, astuple
from typing import List, Any@dataclass
class Book:name: strauthor: str# extend: Any = 'hello world'   # 扩展字段 仅做记录publication: str = field(metadata={"description": "出版日期"})num: int = field(default=1, metadata={"description": "数量"})price: float = field(default=0.0, metadata={'currency': 'RMB'})category: List[str] = field(default_factory=list, repr=False, metadata={"description": "图书类别"})def __post_init__(self):# 方法来确保书名的每个单词的首字母都是大写的self.name = self.name.title()self.title = self.name.upper()def is_expensive(self):# 检查一本书是否价格高于100return self.price > 30.0def total_amount(self):return self.price * self.num
  • 如何调用
    book = Book(name="Under the sky", author="sanxian", price=37.82, publication="2022-11-07", category=['武侠', '古风'])
    book = Book(**{"name": "the three body problem", "author": "刘慈欣", "price": 37.82, "publication": "2022-12-23", "category": ['科幻', '悬疑']})
    print(book.name, book.author)
    print(book.title)print(book.is_expensive())
    print(book.total_amount())book.price = 27.5  # 修改价格属性
    print(book.is_expensive())  # 输出:Trueprint(asdict(book))
    print(astuple(book))
    
    The Three Body Problem 刘慈欣 2022-12-23 37.82
    THE THREE BODY PROBLEM
    True
    37.82
    False
    {'name': 'The Three Body Problem', 'author': '刘慈欣', 'publication': '2022-12-23', 'num': 1, 'price': 27.5, 'category': ['科幻', '悬疑']}
    ('The Three Body Problem', '刘慈欣', '2022-12-23', 1, 27.5, ['科幻', '悬疑'])
    

三、中级用法

  • 装饰器参数 dataclasses.dataclass(*, init=True, repr=True, eq=True, order=False, unsafe_hash=False, frozen=False)

    参数名称默认值是否生成方法说明
    initTrue__init__是否生成 __init__ 方法
    reprTrue__repr__是否生成 __repr__ 方法
    eqTrue__eq__是否生成 __eq__ 方法
    orderFalse<, <=, >, >=是否生成比较方法
    unsafe_hashFalse__hash__是否生成 __hash__ 方法
    frozenFalse不可变类是否生成不可变类
  • init 参数控制是否生成 init 方法。当设置为 True 时,会自动生成 init 方法;当设置为 False 时,不会生成 init 方法。

    @dataclass(init=False)
    class Person:name: strage: intdef __init__(self, name: str, age: int):self.name = nameself.age = agep = Person("zhangsan", 30)
    print(p.name) # 输出: Alice
    print(p.age)  # 输出: 30
    

    init 被设置为 False,dataclass 不会自动生成 init 方法
    init 被设置为 True(即默认),该方法接收两个参数 name 和 age,并将存储在实例属性 self.name 和 self.age 中

  • frozen控制是否创建一个冻结的数据类,即该类的实例在创建后就不可变

    from dataclasses import dataclass@dataclass(frozen=True)
    class Point:x: inty: intp = Point(1, 2)
    p.x = 3  # 会引发 AttributeError: can't set attribute
    

四、其它用法

  • field 支持的参数

    参数描述默认值
    default字段的默认值
    default_factory返回字段初始值的函数
    init是否在._init_()方法中使用字段True
    repr是否在._repr_()方法中使用字段True
    compare是否在比较对象时, 包括该字段True
    hash计算hash时, 是否包括字段True
    metadata包含字段信息的映射
  • compare 默认指定比较,则各个属性依次次进行比较
    注释:下面仅用年龄进行排序或者排序

    @dataclass(order=True)
    class Person:name: str = field(compare=False)age: int = field(compare=True)height: float = field(compare=False)p1 = Person(name="A-zhangsan", age=25, height=156)
    p2 = Person(name="B-lisi", age=18, height=173)
    p3 = Person(name="C-wanger", age=20, height=168)print(p2 > p1)  # 输出 True
    print(sorted([p1,p2, p3]))  # 输出按薪资排序的 Person 对象列表
    
    False
    [Person(name='B-lisi', age=18, height=173), Person(name='C-wanger', age=20, height=168), Person(name='A-zhangsan', age=25, height=156)
    ]
    
http://www.lryc.cn/news/34520.html

相关文章:

  • 【MapGIS精品教程】007:MapGIS投影变换案例教程
  • list数据根据属性字段去重
  • java教程(2023-3-8)
  • node 配置 vue npm配置
  • 特斯拉、小鹏开路,城市NOA距好用还有几年?
  • Vue 3第九章:WatchEffect高级侦听器
  • c++基础——函数
  • DPDK系列之七DPDK中的虚拟化支持
  • 设计模式~桥接模式(bridge)-14
  • Java项目3 电子邮件
  • 设计模式~访问者模式(Visitor)-15
  • 实战小项目之视频监控(1-1)
  • DEJA_VU3D - Cesium功能集 之 103-直角箭头(标绘+编辑)
  • Vue 对象扩展运算符(…)
  • 又是活动 没啥好说的 送代码
  • ARP报文内容详细分析
  • js一键保存当前页面所有图片
  • 【Spring AOP】如何统一“拦截器校验、数据格式返回、异常返回”处理?
  • 规划数据指标体系方法(下)——新海盗模型
  • UML学习备忘录
  • Vue3手写分页在分页的基础上用到Pagination 分页组件
  • 冥想第七百二十四天
  • Jenkins+Docker自动化部署项目
  • TX2配置RealSense D455相机SDK和ros驱动
  • Sentinel架构篇 - 来源访问控制
  • 多线程的Thread 类及方法
  • QT入门Item Views之QTreeView
  • Servlet | Servlet简单入门——构建第一个Servlet项目
  • Spring的IOC/DI,依赖注入的实现
  • 【tensorflow onnx】TensorFlow2导出ONNX及模型可视化教程