深入解析 Python dataclass:类属性与类方法解释

2024-06-21 18:04

本文主要是介绍深入解析 Python dataclass:类属性与类方法解释,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • dataclass
    • 实例属性和类属性
      • 自动设置属性
    • 实例方法
    • 静态方法(@staticmethod)和 类方法(@classmethod)
      • 静态方法
      • 类方法

dataclass

dataclass 是 Python 3.7 引入的一个装饰器,用于简化类的定义。

使用 dataclass 可以自动生成一些常用方法,比如 __init____repr____eq__ 等等。这些方法均无需手动实现,除非有定制化需求。

本文介绍 dataclass 的用法,同时也介绍了相关面向对象的一些属性和方法,包括:

  1. 如何判断 实例属性和类属性;
  2. 对比分析 实例方法、静态方法(@staticmethod)和 类方法(@classmethod

实例属性和类属性

使用 dataclass 定义一个类非常简单,只需要在类定义前加上 @dataclass 装饰器,并在类中定义属性即可。

from dataclasses import dataclass, field@dataclass
class Person:home = "Wuhan"name : str

类属性是定义在类体中,在任何方法之外(包括__init__方法)的变量。它们属于类本身,而不是类的实例。类属性在内存中只有一个副本,无论创建了多少个类的实例,这个副本都是共享的。

在上述的代码中,home就是一个类属性。它被定义在类体中,在任何方法之外,且没有与实例变量相同的类型注解。由于home被定义为一个字符串,它将作为Person类的一个共享属性。所有Person类的实例都将访问到同一个home属性的值。

实例属性 是定义在类的__init__方法或其他实例方法中的变量。它们属于类的实例,每个实例都有自己的实例属性副本。实例属性通常通过类型注解来指定其期望的数据类型,并使用self参数来引用实例本身。

在上述代码中,name被声明为一个类型注解为str的变量,虽然其没有在__init__方法内部被初始化。由于使用了@dataclass装饰器,dataclasses模块会自动生成__init__方法,该方法将接受与类型注解相匹配的参数,并将它们设置为实例属性。因此,尽管没有明确写出__init__方法,但name仍然是一个实例属性。

故在创建一个Person类的实例时,需要提供一个name值,该值将被设置为该实例的name属性。由于name是一个实例属性,每个Person实例都将有自己的name属性副本。只需要给实例属性 name 传参,无需给类属性home传参。

p1 = Person("Alice")
p1 = Person("Bob")
print(p1.name, p1.home)
print(p2.name, p2.home)

Output:

Alice Wuhan
Bob Wuhan

由于home是类属性,故对其进行修改后,p1.homep2.home会变成同一个值。

Person.home = "Fuzhou"
print(p1.home, p2.home)

Output:

Fuzhou Fuzhou

自动设置属性

在初始化某些类别的时候,某些属性,可以通过传入的属性,推测与计算出来。

dataclass 中,实例变量是在 __init__ 方法中初始化的属性。可以使用 field 函数并指定 init=False,表示这个字段不在 __init__ 方法中进行初始化。

以下述代码为例,根据输入的年龄 age,添加是否成年( is_adult) 的属性。__post_init__ __init__之后运行,在其中实现为Person 实例,添加 is_adult 的属性。

在下面的例子中,is_adult 的默认值是 False,并且不会在 __init__ 方法中初始化。通过定义一个 __post_init__ 方法,对 is_adult进行赋值。

@dataclass
class Person:name : strage: intis_adult: bool = field(init=False, default=False)def __post_init__(self):if self.age >= 18:self.is_adult = Trueelse:self.is_adult = False
p1 = Person("Alice", 25)
p2 = Person("Bob", 17)
print(p1)
print(p2)

Output:

Person(name='Alice', age=25, is_adult=True)
Person(name='Bob', age=17, is_adult=False)

如结果所示,实现自动根据 age,为Person 实例,添加 is_adult 属性。

实例方法

在面向对象编程(Object-Oriented Programming, OOP)的术语中,一个实例方法 能够访问和改变对象状态。

@dataclass
class Person:name : strage: intdef grow_up(self):self.age += 1

grow_up 是一个实例方法(Instance Method)定义在 Person 类中。

静态方法(@staticmethod)和 类方法(@classmethod)

除了实例方法之外,还有两种类型的方法:静态方法(@staticmethod)和类方法(@classmethod)。它们的用途和定义方式如下:

  1. 静态方法与类或实例无关,它只是类定义体中的一个普通函数。静态方法不需要传递类实例(self)或类(cls)作为第一个参数。

  2. 类方法是绑定到类而不是类实例的方法。类方法的第一个参数是类本身(通常命名为 cls)。类方法可以访问和修改类状态。

静态方法

静态方法,比较简单且易于理解,不做过多描述;

写一个简单的静态方法 ,供大家参考!

@dataclass
class Person:name : strage: int@staticmethoddef add(value):return value + 1def grow_up(self):# self.age  = self.add(self.age)self.age  = Person.add(self.age)

add 是一个静态方法,使用self.addPerson.add 都可以。

p = Person("Alice", 17)
p.grow_up()
print(p)

Output:

Person(name='Alice', age=18)

类方法

在类方法中,使用 cls 而不是 self

在Python中,定义一个类方法(即使用@classmethod装饰器的方法)时,第一个参数被命名为cls而不是self

cls是一个约定俗成的名称 用来代表类本身,而不是类的实例。

使用cls而不是self的原因是:

  1. 语义清晰self通常用于表示类的实例,而**cls用于表示类本身**。使用不同的名称可以帮助阅读代码的人更容易地理解方法的上下文和用途。
  2. 避免混淆:在类方法中,实际上并没有一个类的实例可以引用,因此使用self可能会引发混淆。如果尝试在类方法内部使用self作为第一个参数,Python解释器会抛出一个TypeError,因为它期望的是类本身,而不是类的实例。
  3. 符合Python的习惯和约定:Python社区广泛接受并使用cls作为类方法的第一个参数的命名约定。这有助于保持代码的一致性和可读性。

下面是一个简单的例子,展示了为什么类方法需要cls而不是self

@dataclass
class Person:home = "Wuhan"name : str@classmethoddef get_name(cls) -> str:return cls.name@classmethoddef get_home(cls) -> str:return cls.homep = Person("Alice")
p.get_home()

Output:

'Wuhan'

类方法传入的 cls,是类本身而不是类的实例化。

如果忘记了,为什么 home 是 类属性,name 是实例属性,请返回 实例属性和类属性 章节重新阅读!

由于 home 是类属性,cls 可以访问到;name 是实例属性,cls 无法访问到,只有实例对象 (self) 才能访问到 name。

所以 get_home 成功, get_name 失败。

p.get_name()

Output:
在这里插入图片描述

这篇关于深入解析 Python dataclass:类属性与类方法解释的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1081934

相关文章

Nginx 访问控制的多种方法

《Nginx访问控制的多种方法》本文系统介绍了Nginx实现Web访问控制的多种方法,包括IP黑白名单、路径/方法/参数控制、HTTP基本认证、防盗链机制、客户端证书校验、限速限流、地理位置控制等基... 目录一、IP 白名单与黑名单1. 允许/拒绝指定IP2. 全局黑名单二、基于路径、方法、参数的访问控制

Mybatis的mapper文件中#和$的区别示例解析

《Mybatis的mapper文件中#和$的区别示例解析》MyBatis的mapper文件中,#{}和${}是两种参数占位符,核心差异在于参数解析方式、SQL注入风险、适用场景,以下从底层原理、使用场... 目录MyBATis 中 mapper 文件里 #{} 与 ${} 的核心区别一、核心区别对比表二、底

Python中Request的安装以及简单的使用方法图文教程

《Python中Request的安装以及简单的使用方法图文教程》python里的request库经常被用于进行网络爬虫,想要学习网络爬虫的同学必须得安装request这个第三方库,:本文主要介绍P... 目录1.Requests 安装cmd 窗口安装为pycharm安装在pycharm设置中为项目安装req

Python容器转换与共有函数举例详解

《Python容器转换与共有函数举例详解》Python容器是Python编程语言中非常基础且重要的概念,它们提供了数据的存储和组织方式,下面:本文主要介绍Python容器转换与共有函数的相关资料,... 目录python容器转换与共有函数详解一、容器类型概览二、容器类型转换1. 基本容器转换2. 高级转换示

nginx跨域访问配置的几种方法实现

《nginx跨域访问配置的几种方法实现》本文详细介绍了Nginx跨域配置方法,包括基本配置、只允许指定域名、携带Cookie的跨域、动态设置允许的Origin、支持不同路径的跨域控制、静态资源跨域以及... 目录一、基本跨域配置二、只允许指定域名跨域三、完整示例四、配置后重载 nginx五、注意事项六、支持

MySQL查看表的历史SQL的几种实现方法

《MySQL查看表的历史SQL的几种实现方法》:本文主要介绍多种查看MySQL表历史SQL的方法,包括通用查询日志、慢查询日志、performance_schema、binlog、第三方工具等,并... 目录mysql 查看某张表的历史SQL1.查看MySQL通用查询日志(需提前开启)2.查看慢查询日志3.

MySQL底层文件的查看和修改方法

《MySQL底层文件的查看和修改方法》MySQL底层文件分为文本类(可安全查看/修改)和二进制类(禁止手动操作),以下按「查看方法、修改方法、风险管控三部分详细说明,所有操作均以Linux环境为例,需... 目录引言一、mysql 底层文件的查看方法1. 先定位核心文件路径(基础前提)2. 文本类文件(可直

Java实现字符串大小写转换的常用方法

《Java实现字符串大小写转换的常用方法》在Java中,字符串大小写转换是文本处理的核心操作之一,Java提供了多种灵活的方式来实现大小写转换,适用于不同场景和需求,本文将全面解析大小写转换的各种方法... 目录前言核心转换方法1.String类的基础方法2. 考虑区域设置的转换3. 字符级别的转换高级转换

使用Python将PDF表格自动提取并写入Word文档表格

《使用Python将PDF表格自动提取并写入Word文档表格》在实际办公与数据处理场景中,PDF文件里的表格往往无法直接复制到Word中,本文将介绍如何使用Python从PDF文件中提取表格数据,并将... 目录引言1. 加载 PDF 文件并准备 Word 文档2. 提取 PDF 表格并创建 Word 表格

使用Python实现局域网远程监控电脑屏幕的方法

《使用Python实现局域网远程监控电脑屏幕的方法》文章介绍了两种使用Python在局域网内实现远程监控电脑屏幕的方法,方法一使用mss和socket,方法二使用PyAutoGUI和Flask,每种方... 目录方法一:使用mss和socket实现屏幕共享服务端(被监控端)客户端(监控端)方法二:使用PyA