【Python 高级特性】深入 NamedTuple 命名元组

2023-10-30 06:04

本文主要是介绍【Python 高级特性】深入 NamedTuple 命名元组,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

介绍

和元组 tuple 一样,NamedTuple 也是不可变数据类型,创建之后就不能改变内容。
如其名,和 tuple 的区别在于“Named”,即"命名"。NamedTuple 不像数组那样使用下标读写,反而和类相似,使用 . 来读写。

基本语法

创建 NamedTuple 的函数定义

collections.namedtuple(typename, field_names, *, rename=False, defaults=None, module=None)

参数说明:

  • typename:新创建的类的名称。
  • field_names:字段名称列表。必须是有效的 Python 变量名称,且不能以下划线开头
  • rename:是否自动转换无效字段名。
  • defaults:字段默认值列表。
  • module__module__ 的值。

使用教程

创建

首先看看如何创建命名元组。以 Point(代表二维坐标中的一个点)为例:

# 导包
from collections import namedtuple# 创建普通元组
point = (22, 33)
print(point) # 输出:(22, 33)# 创建命名元组
Point = namedtuple('Point', 'x y')
point_A = Point(22, 33)
print(point_A) # 输出:Point(x=22, y=33)

重点是这两句话

Point = namedtuple('Point', 'x y')
point_A = Point(22, 33)

需要注意,namedtuple() 是用来创建类的,不是创建对象实例!

我们先用 namedtuple 创建了一个名为 Point,有两个字段 xy 的子类,然后将这个类赋给 Point 变量。
然后 Point(22, 33) 就是普通的 new 的语法。

类似于如下代码:

class Point:def __init__(self, x, y):self.x = xself.y = y
point_A = Point(22, 33)

创建命名元组对象时,也可以使用位置参数

a = Point(1, 2)
b = Point(y=2, x=1)
a == b # >>> True

field_names 参数用来设置命名元组字段名,有三种风格可以选择。
下面几种都是等价写法:

Point = namedtuple('Point', 'x y')
Point = namedtuple('Point', 'x,y')
Point = namedtuple('Point', ['x', 'y'])# 下面都是合法代码
# 中间允许存在任意空白字符
Point = namedtuple('Point', 'x,   \t\t\t\n\n y')
Point = namedtuple('Point', 'x   \t\t\t\n\n y')
# 元组也可以
Point = namedtuple('Point', ('x', 'y'))
# 事实上只要是可迭代都行
def fields():yield 'x'yield 'y'
Point = namedtuple('Point', fields())

使用

命名元组首先是一个元组,元组能怎么用,命名元组当然也可以。

print(point_A[0])
print(point_A[1])
print(*point_A) # tuple unpack# 输出
"""
22
33
22 33
"""

然后是命名元组的特殊用法:

print(point_A.x)
print(point_A.y)# 输出
"""
22
33
"""

常用方法

namedtuple 创建的类还附赠了一些实用方法:

Point._make(iterable) # 从某个序列创建命名元组
point._asdict() # 转成字典
point._replace(**kargs) # 返回一个新元组,新元组里的指定字段被替换为指定值point._fields # 列出字段名
point._field_defaults # 列出字段默认值

设置默认值

可以为命名元组的字段设置默认值,只需要在创建类的时候传入 defaults 参数即可。

# 四维向量
# 默认值为 Vector4D(0, 0, 0, 0)
Vector4 = namedtuple('Vector4D', 'x y z w', defaults=(0, 0, 0, 0))v1 = Vector4()
v2 = Vector4(1)
v3 = Vector4(1, 2, w=4)
print(v1)
print(v2)
print(v3)# 输出
"""
Vector4D(x=0, y=0, z=0, w=0)
Vector4D(x=1, y=0, z=0, w=0)
Vector4D(x=1, y=2, z=0, w=4)
"""

默认值的数量可以小于字段数,表示为右边 n 个参数设置默认值。

Foo = namedtuple('Foo', 'a b c d', defaults=(1, 2))
print(Foo(22, 33))
print(Foo())# 输出
"""
Foo(a=22, b=33, c=1, d=2)
Traceback (most recent call last):File "D:\TempCodeFiles\named_tuple.py", line 6, in <module>print(Foo())
TypeError: Foo.__new__() missing 2 required positional arguments: 'a' and 'b'
"""

更好的表示方式

namedtuple() 的写法既不直观,也不优雅。Python 3.5 新增了一种更好的写法:

# >= Python 3.5
from typing import NamedTuple
class PointA(NamedTuple):x: int = 0y: int = 0# >= Python 2
from collections import namedtuple
PointB = namedtuple('PointB', 'x y', defaults=(0, 0))print(PointA(2, 3) == PointB(2, 3)) # 输出:True

继承并扩展 NamedTuple

namedtuple() 返回的是一个正常的类。既然它是一个类,当然也可以被继承。

创建一个 Point 命名元组,增加一个方法,求两点距离。

# >= Python 3.5
class Point(NamedTuple):x: int = 0y: int = 0def distance(self, p) -> float:return math.sqrt((self.x - p.x) ** 2 + (self.y - p.y) ** 2)# >= Python 2
class Point(namedtuple('Point', 'x y', defaults=(0, 0))):def distance(self, p) -> float:return math.sqrt((self.x - p.x) ** 2 + (self.y - p.y) ** 2)a = Point()
b = Point(3, 2)
print(a, b)
print(a.distance(b))

应用

读 csv 文件

以读入一个储存英语单词的 csv 文件为例。

import csv
from collections import namedtuple# 定义命名元组
# 按照 csv 列名来定义字段
Word = namedtuple('Word', 'word, type, chs_def, eng_ch, context, example')file_path = r'C:\Users\ZhouXiaokang\Desktop\单词 Vol 1 Ch 1 Ep 2.csv'
with open(file_path, 'r', encoding='utf-8') as f:reader = csv.reader(f)next(reader) # 跳过标题行for word in map(Word._make, reader):print(f'{word.word} {word.type}. {word.chs_def} | 例:{word.context}')

输出

chirp n&v. (鸟、昆虫)啾啾叫,发唧唧声 | 例:(*chirp* *chirp* *chirp*)
screech v. (车辆、汽车轮胎)发出刺耳声 | 例:(*screech*)
Shiroko term. 白子 | 例:
mug v. 对…行凶抢劫 | 例:You didn't get mugged, did you?
faint v. 晕厥;晕倒 | 例:What's that? You fainted from hunger?
......

作为字典的代替品表示数据

相对于字典的优势:
1.快、小
2..field['field'] 更清晰

以下源码摘自 baidupcs_py 库:

class PcsFile(NamedTuple):"""A Baidu PCS filepath: str  # remote absolute pathis_dir: Optional[bool] = Noneis_file: Optional[bool] = Nonefs_id: Optional[int] = None  # file idsize: Optional[int] = Nonemd5: Optional[str] = Noneblock_list: Optional[List[str]] = None  # block md5 listcategory: Optional[int] = Noneuser_id: Optional[int] = Nonectime: Optional[int] = None  # server created timemtime: Optional[int] = None  # server modifed timelocal_ctime: Optional[int] = None  # local created timelocal_mtime: Optional[int] = None  # local modifed timeserver_ctime: Optional[int] = None  # server created timeserver_mtime: Optional[int] = None  # server modifed timeshared: Optional[bool] = None  # this file is shared if True"""path: str  # remote absolute pathis_dir: Optional[bool] = Noneis_file: Optional[bool] = Nonefs_id: Optional[int] = None  # file idsize: Optional[int] = Nonemd5: Optional[str] = Noneblock_list: Optional[List[str]] = None  # block md5 listcategory: Optional[int] = Noneuser_id: Optional[int] = Nonectime: Optional[int] = None  # server created timemtime: Optional[int] = None  # server modifed timelocal_ctime: Optional[int] = None  # local created timelocal_mtime: Optional[int] = None  # local modifed timeserver_ctime: Optional[int] = None  # server created timeserver_mtime: Optional[int] = None  # server modifed timeshared: Optional[bool] = None  # this file is shared if Truerapid_upload_info: Optional[PcsRapidUploadInfo] = Nonedl_link: Optional[str] = None@staticmethoddef from_(info) -> "PcsFile":return PcsFile(path=info.get("path"),is_dir=info.get("isdir") == 1,is_file=info.get("isdir") == 0,fs_id=info.get("fs_id"),size=info.get("size"),md5=info.get("md5"),block_list=info.get("block_list"),category=info.get("category"),user_id=info.get("user_id"),ctime=info.get("ctime"),mtime=info.get("mtime"),local_ctime=info.get("local_ctime"),local_mtime=info.get("local_mtime"),server_ctime=info.get("server_ctime"),server_mtime=info.get("server_mtime"),shared=info.get("shared"),)

源码

见 Github。

关键部分在这里:

    # Build-up the class namespace dictionary# and use type() to build the result class# 收集类的方法、字段等class_namespace = {'__doc__': f'{typename}({arg_list})','__slots__': (),'_fields': field_names,'_field_defaults': field_defaults,'__new__': __new__,'_make': _make,'__replace__': _replace,'_replace': _replace,'__repr__': __repr__,'_asdict': _asdict,'__getnewargs__': __getnewargs__,'__match_args__': field_names,}for index, name in enumerate(field_names):doc = _sys.intern(f'Alias for field number {index}')class_namespace[name] = _tuplegetter(index, doc)# 创建新类result = type(typename, (tuple,), class_namespace) 

type() 函数传入一个参数,用来获取对象的类;如果传入三个参数,就变成了动态创建类,相当于 class 的动态写法。

class Foo:def hello(self):print('Hello')
# 等价于
def hello(self):print('Hello')
Foo = type('Foo', (object,), {'hello': hello})

参考文章

  1. https://docs.python.org/zh-cn/3/library/collections.html#collections.namedtuple
  2. https://realpython.com/python-namedtuple/

这篇关于【Python 高级特性】深入 NamedTuple 命名元组的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

python: 多模块(.py)中全局变量的导入

文章目录 global关键字可变类型和不可变类型数据的内存地址单模块(单个py文件)的全局变量示例总结 多模块(多个py文件)的全局变量from x import x导入全局变量示例 import x导入全局变量示例 总结 global关键字 global 的作用范围是模块(.py)级别: 当你在一个模块(文件)中使用 global 声明变量时,这个变量只在该模块的全局命名空

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

深入探索协同过滤:从原理到推荐模块案例

文章目录 前言一、协同过滤1. 基于用户的协同过滤(UserCF)2. 基于物品的协同过滤(ItemCF)3. 相似度计算方法 二、相似度计算方法1. 欧氏距离2. 皮尔逊相关系数3. 杰卡德相似系数4. 余弦相似度 三、推荐模块案例1.基于文章的协同过滤推荐功能2.基于用户的协同过滤推荐功能 前言     在信息过载的时代,推荐系统成为连接用户与内容的桥梁。本文聚焦于

变量与命名

引言         在前两个课时中,我们已经了解了 Python 程序的基本结构,学习了如何正确地使用缩进来组织代码,并且知道了注释的重要性。现在我们将进一步深入到 Python 编程的核心——变量与命名。变量是我们存储数据的主要方式,而合理的命名则有助于提高代码的可读性和可维护性。 变量的概念与使用         在 Python 中,变量是一种用来存储数据值的标识符。创建变量很简单,

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

【学习笔记】 陈强-机器学习-Python-Ch15 人工神经网络(1)sklearn

系列文章目录 监督学习:参数方法 【学习笔记】 陈强-机器学习-Python-Ch4 线性回归 【学习笔记】 陈强-机器学习-Python-Ch5 逻辑回归 【课后题练习】 陈强-机器学习-Python-Ch5 逻辑回归(SAheart.csv) 【学习笔记】 陈强-机器学习-Python-Ch6 多项逻辑回归 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch7 判别分析 【学

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识

nudepy,一个有趣的 Python 库!

更多资料获取 📚 个人网站:ipengtao.com 大家好,今天为大家分享一个有趣的 Python 库 - nudepy。 Github地址:https://github.com/hhatto/nude.py 在图像处理和计算机视觉应用中,检测图像中的不适当内容(例如裸露图像)是一个重要的任务。nudepy 是一个基于 Python 的库,专门用于检测图像中的不适当内容。该

【C++高阶】C++类型转换全攻略:深入理解并高效应用

📝个人主页🌹:Eternity._ ⏩收录专栏⏪:C++ “ 登神长阶 ” 🤡往期回顾🤡:C++ 智能指针 🌹🌹期待您的关注 🌹🌹 ❀C++的类型转换 📒1. C语言中的类型转换📚2. C++强制类型转换⛰️static_cast🌞reinterpret_cast⭐const_cast🍁dynamic_cast 📜3. C++强制类型转换的原因📝