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

符合Python风格的对象(覆盖类属性)

覆盖类属性

Python 有个很独特的特性:类属性可用于为实例属性提供默认
值。Vector2d 中有个 typecode 类属性,bytes 方法两次用到了
它,而且都故意使用 self.typecode 读取它的值。因为 Vector2d 实
例本身没有 typecode 属性,所以 self.typecode 默认获取的是
Vector2d.typecode 类属性的值。

但是,如果为不存在的实例属性赋值,会新建实例属性。假如我们为
typecode 实例属性赋值,那么同名类属性不受影响。然而,自此之
后,实例读取的 self.typecode 是实例属性 typecode,也就是把同
名类属性遮盖了。借助这一特性,可以为各个实例的 typecode 属性定
制不同的值。

Vector2d.typecode 属性的默认值是 ‘d’,即转换成字节序列时使用
8 字节双精度浮点数表示向量的各个分量。如果在转换之前把
Vector2d 实例的 typecode 属性设为 ‘f’,那么使用 4 字节单精度浮
点数表示各个分量,如示例 9-13 所示。
我们在讨论如何添加自定义的实例属性,因此示例 9-13 使
用的是示例 9-9 中不带__slots__ 属性的 Vector2d 类。
示例 9-13 设定从类中继承的 typecode 属性,自定义一个实例属性

>>> from vector2d_v3 import Vector2d
>>> v1 = Vector2d(1.1, 2.2)
>>> dumpd = bytes(v1)
>>> dumpd
b'd\x9a\x99\x99\x99\x99\x99\xf1?\x9a\x99\x99\x99\x99\x99\x01@'
>>> len(dumpd) # ➊
17
>>> v1.typecode = 'f' # ➋
>>> dumpf = bytes(v1)
>>> dumpf
b'f\xcd\xcc\x8c?\xcd\xcc\x0c@'
>>> len(dumpf) # ➌
9 >>> Vector2d.typecode #'d'

❶ 默认的字节序列长度为 17 个字节。
❷ 把 v1 实例的 typecode 属性设为 ‘f’。
❸ 现在得到的字节序列是 9 个字节长。
❹ Vector2d.typecode 属性的值不变,只有 v1 实例的 typecode 属
性使用 ‘f’。
现在你应该知道为什么要在得到的字节序列前面加上 typecode 的值
了:为了支持不同的格式。
如果想修改类属性的值,必须直接在类上修改,不能通过实例修改。如
果想修改所有实例(没有 typecode 实例变量)的 typecode 属性的默
认值,可以这么做:

>>> Vector2d.typecode = 'f'

然而,有种修改方法更符合 Python 风格,而且效果持久,也更有针对
性。类属性是公开的,因此会被子类继承,于是经常会创建一个子类,
只用于定制类的数据属性。Django 基于类的视图就大量使用了这个技
术。具体做法如示例 9-14 所示。

示例 9-14 ShortVector2d 是 Vector2d 的子类,只用于覆盖
typecode 的默认值

>>> from vector2d_v3 import Vector2d
>>> class ShortVector2d(Vector2d): # ➊
... typecode = 'f'
...
>>> sv = ShortVector2d(1/11, 1/27) # ➋
>>> sv
ShortVector2d(0.09090909090909091, 0.037037037037037035) # ➌
>>> len(bytes(sv)) # ➍
9

❶ 把 ShortVector2d 定义为 Vector2d 的子类,只用于覆盖
typecode 类属性。
❷ 为了演示,创建一个 ShortVector2d 实例,即 sv。
❸ 查看 sv 的 repr 表示形式。
❹ 确认得到的字节序列长度为 9 字节,而不是之前的 17 字节。
这也说明了我在

Vecto2d.__repr__ 

方法中为什么没有硬编码
class_name 的值,而是使用

type(self).__name__ 

获取,如下所
示:

# 在Vector2d类中定义
def __repr__(self):class_name = type(self).__name__return '{}({!r}, {!r})'.format(class_name, *self)

如果硬编码 class_name 的值,那么 Vector2d 的子类(如
ShortVector2d)要覆盖__repr__ 方法,只是为了修改 class_name
的值。从实例的类型中读取类名__repr__ 方法就可以放心继承。

至此,我们通过一个简单的类说明了如何利用数据模型处理 Python 的其
他功能:提供不同的对象表示形式、实现自定义的格式代码、公开只读
属性,以及通过 hash() 函数支持集合和映射。

http://www.lryc.cn/news/2391925.html

相关文章:

  • 题目 3314: 蓝桥杯2025年第十六届省赛真题-魔法科考试
  • Java八股-Java优缺点,跨平台,jdk、jre、jvm关系,解释和编译
  • linux 内核态和用户态定时器函数使用总结
  • 支持selenium的chrome driver更新到136.0.7103.113
  • 硬件服务器基础
  • LVS的DR模式部署
  • TRS收益互换平台开发实践:从需求分析到系统实现
  • 测试Bug篇
  • 【Linux系统移植】Cortex-A8 Linux系统移植(超详细)
  • 第十五届蓝桥杯大赛软件赛国赛Python 大学 C 组试做【本期题单: 设置密码、栈】
  • 报错SvelteKitError: Not found: /.well-known/appspecific/com.chrome.devtools.json
  • word添加页眉
  • JavaScript 中的 BigInt:当普通数字不够“大“时的救星
  • 通过mailto:实现web/html邮件模板唤起新建邮件并填写内容
  • 2025推客系统小程序开发:独立部署源码交付,高性价比裂变增长引擎
  • 【Python零基础入门系列】第6篇:Python 的数据结构世界(列表、字典、集合、元组)
  • MVCC 原理与并发控制实现
  • CentOS 7.0重置root密码
  • YOLOV11改进策略【最新注意力机制】CVPR2025局部区域注意力机制LRSA-增强局部区域特征之间的交互
  • 3D建模的全景图谱:从55个工具到元宇宙的数字革命
  • Kotlin 活动事件通讯跳转深度讲解
  • vue + ant-design + xlsx 实现Excel多Sheet页导出功能
  • 关于 Web 安全:6. 常见 CMS 开源系统风险点
  • DAY33 简单神经网络
  • OBOO鸥柏丨2025年鸿蒙生态+国产操作系统触摸屏查询一体机核心股
  • 【观成科技】Ymir勒索软件组织窃密木马RustyStealer加密通信分析
  • Vuer开源程序 是一个轻量级的可视化工具包,用于与动态 3D 和机器人数据进行交互。它支持 VR 和 AR,可以在移动设备上运行。
  • 浅谈学习(费曼学习法)
  • 高光谱成像相机:表型技术在林业育种和精确林业的应用
  • iOS App启动优化(冷启动、热启动)