深入解析 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:
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/34480.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Django教程(001):安装及快速上手

1.1 Django安装 pip install django安装之后 c:\python39-python.exe-Scripts-pip.exe-django-admin.exe【安装django之后,工具,创建django项目】-Lib-内置模块-site-packages-flask-django(安装django之后,【django框架源码】)如下图&…

思考题:相交的几何图形

给定不超过 26 个几何图形,每个图形都有一个唯一大写字母作为其编号。 每个图形在平面中的具体位置已知,请你判断,对于每个图形,有多少个其他图形与其存在交点。 在判断交点时,只考虑边与边相交的情况,如…

AIGC+艺术=教育变革?

在数字化时代的浪潮中,技术的每一次跃进都深刻影响着社会的各个领域,教育亦不例外。近年来,人工智能生成内容(AIGC)技术的兴起,为艺术教育领域带来了前所未有的变革机遇。当AIGC与艺术相结合,我…

vscode 删除不用的ssh远程连接

使用vscode连接一个远程服务器发现联不通,但是使用mobaxterm是可以通的,最后原因发现是这个服务器ip与之前连过的另一台相同,和之前连接保存的信息冲突了 解决办法: 使用记事本打开这个路径下的known_hosts(最好备份一下)&#x…

电脑打印文件怎么操作?

有打印机用户的打印操作 对于已经拥有打印机的用户来说,打印文件通常是一个简单的步骤。首先,你需要将你的文件(如Word、PDF、PPT等)在电脑上打开。然后,点击菜单栏中的“打印”选项,或者快捷键CtrlP&…

CRMEB-PHP多商户版安装系统配置清单

系统在安装完成之后,需要对系统进行一系列的配置,才能正常使用全部的功能,以下是官方整理的配置清单 平台后台 商户后台

实例080 进度条百分比显示

本文仅供学习交流,严禁用于商业用途,如本文涉及侵权请及时联系本人将于及时删除 目录 1.实例说明 2.技术要点 3.实现过程 4.实例结果 5.示例拓展 2.10 进度条控件典型实例进度条控件(Progress)用于显示程序的进度&#xff0c…

NetSuite 文件夹 Group Restriction的探究

同一个角色,为什么相同的文件,有的用户可以看,而有的用户不能看呢?这其中与一个隐藏功能相关,即文件夹的Restriction相关,其中一个非常典型的点是Group Restriction(组限制)&#xf…

Web渗透:XXE-XML外部实体漏洞

XML External Entity (XXE) 漏洞是一种注入攻击,利用不安全的XML解析器来执行各种恶意操作,如读取本地文件、执行远程代码、发起拒绝服务攻击等;此漏洞的根本原因在于XML标准允许在文档中定义外部实体,并在解析时进行解析和替换。…

Navicat连接服务器MySQL

Navicat连接服务器MySQL 1. Navicat连接服务器MySQL2. 如何查看MySQL用户名和密码3. 修改MySQL登录密码4. 安装MySQL(Centos7)遇到错误和问题1. error 1045 (28000): access denied for user rootlocalhost (using password:yes) 1. Navicat连接服务器MySQL 选择数据库 直接使用…

jar包转exe封装软件并一键安装使用

目录 封装文档说明如有跨域问题在ems服务增加配置服务启动自动打开浏览器使用工具 Launch4j 把jar包打成exe执行文件现在还不能给用户用,因为缺少jre,后面整合资源会把jre一起打包使用inno setup合并资源,mysql之类的服务,最终打包…

【系统架构设计师】四、嵌入式基础知识(软件|软件设计|硬件|式总线逻辑)

目录 一、嵌入式软件 1.1 嵌入式软件分类 1.2 板级支持包(BSP) 1.3 BootLoader 1.4 设备驱动程序 二、嵌入式软件设计 2.1 编码 2.2 交叉编译 2.3 交叉调试 三、嵌入式系统硬件的分类 3.1 根据用途分类 3.2 存储器分类 四、内(外)总线逻辑 …

uni-app系列:uni.navigateTo传值跳转

文章目录 1. 使用URL参数2. 使用页面栈注意事项:uni.navigateTo API 参数详细说明回调函数参数 在uni-app中,如果想要通过uni.navigateTo方法跳转到另一个页面并传递参数,可以使用页面路由的URL参数或者页面栈的方式来传递。但是,…

分享一个解决 EF 性能低的思路,通过 Python 访问心跳侦测 API 保持 EF 在线

前言 .NET FrameWork EF(Entity Framework)是一个很优秀的 ORM 框架,对于提高工作效率很有帮助,但由于底层原因,在程序启动时,EF 需要初始化和加载模型及其对应的数据库元数据等等,涉及到数据库…

20240507-招商证券 基于鳄鱼线的指数择时及轮动策略

动量震荡指标构造 动量震荡指标为交易者提供了获利的钥匙。动量震荡指标测算了5根价格柱相对于34根价格柱的动量变化。首先计算最近5根价格柱的最高价和最低价间的中点的简单移动平均值,即(最高价最低价)12的简单移动平均,将得出的值减去最近34根价格柱…

Cephalo:专门用于仿生设计的多模态视觉大型语言模型

实时了解业内动态,论文是最好的桥梁,专栏精选论文重点解读热点论文,围绕着行业实践和工程量产。若在某个环节出现卡点,可以回到大模型必备腔调或者LLM背后的基础模型重新阅读。而最新科技(Mamba,xLSTM,KAN)…

审美进阶:7个小程序模板,助你提高设计感!

小程序是一种无需下载和安装即可使用的应用程序。小程序实现了应用程序“触手可及”的梦想。用户可以通过扫描或搜索打开应用程序。对于开发者来说,小程序也大大降低了开发成本。因此,越来越多的品牌争相制作小程序应用程序。本文将为您带来优秀的微信小…

ubuntu安装qtcreator与环境配置

sudo apt-get update sudo apt-get install gcc g # 两个编译器 sudo apt-get install build-essential # 编译c/c所需要的软件包 sudo apt-get install libgl1-mesa-dev # 安装mesa,Mesa 实际上是一个库,它实现了多种图形 API 规范 sudo apt-get insta…

MFC结构体定义中使用CString类型

MFC的结构体定义中字符串可以使用CString类型; struct MyStruct {int x;int y;CString str; }; ...... void CTttView::OnDraw(CDC* pDC) {CTttDoc* pDoc GetDocument();ASSERT_VALID(pDoc);// TODO: add draw code for native data hereCString str1;MyStruct m_…

vue配置中的process.env

项目中的.env开头的文件是否知道是干什么的呢 主要是为了区分测试环境还是生产环境env.development为测试环境 # 测试环境 NODE_ENV development VUE_APP_BASE_API http://xxxxxxxxx // 命名一定要以 VUE_APP_ 开头,要不然根本取不到 .env.production为生产环境…