如何在 Python 中创建一个不可变的字典 - Adam Johnson

liftword4个月前 (03-07)技术文章25

Python 的内置集合类型具有可变和不可变两种风格:

  • 可变版本:list 不可变版本:tuple
  • 可变版本:set 不可变版本:frozenset
  • 可变版本:dict 不可变版本:MappingProxyType

PEP 416 早在 2012 年就为 Python 3.3提出了一种类型frozendict。 PEP 被拒绝,原因有很多。推理包括几个关于不可变 dict 的效用的问题,在将它们添加到代码之前值得检查一下。

但是 PEP确实为我们提供了一个模拟不可变dicts:的工具 types.MappingProxyType 。此类型是 adict或其他映射的只读代理。Python 在内部将这种类型用于重要的词典,这就是为什么您不能随意修补内置类型的原因。

Python 3.3 中唯一的变化是为用户代码公开这种类型。

要创建一个“不可变”的字典,MappingProxyType从字典中创建一个,而不保留对底层字典的任何引用:

from types import MappingProxyType

power_levels = MappingProxyType(
    {
        "Kevin": 9001,
        "Benny": 8000,
    }
)
读取:
In [1]: power_levels["Kevin"]
Out[1]: 9001

In [2]: power_levels["Benny"]
Out[2]: 8000

In [3]: list(power_levels.keys())
Out[3]: ['Kevin', 'Benny']

但是,任何更改值的尝试都将导致TypeError:

In [4]: power_levels["Benny"] = 9200
---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
 in 
----> 1 power_levels["Benny"] = 9200

TypeError: 'mappingproxy' object does not support item assignment

要使用更改创建映射代理的副本,您可以使用 Python 3.9 的 dict 合并运算符 。将映射代理与新字典合并,并将结果传递给MappingProxyType

In [10]: benny_better = MappingProxyType(power_levels | {"Benny": 9200})

In [11]: benny_better
Out[11]: mappingproxy({'Kevin': 9001, 'Benny': 9200})

对于更复杂的修改,您可以将映射代理复制到新的dict进行更改,然后将结果转换为映射代理:

In [12]: new_world = power_levels | {}

In [13]: del new_world["Benny"]

In [14]: del new_world["Kevin"]

In [15]: new_world["Bock"] = 100

In [16]: new_world = MappingProxyType(new_world)

In [17]: new_world
Out[17]: mappingproxy({'Bock': 100})

有几个提供不可变数据结构的第三方包,例如 immutables 和 pyrsistent 。这些 API 稍微好一些,但会带来不同的权衡,例如性能和维护状态。如果MappingProxyType不适合您,您可能想研究它们,但我鼓励尽可能多地使用标准库。

相关文章

探索 Python 中合并两个字典的七种方法,找到最适合你的那一款!

在 Python 的世界里,字典作为一种强大的数据结构,扮演着举足轻重的角色。而合并字典,则是我们经常遇到的操作。Python 为我们提供了多种合并字典的方式,每种方式都有其独特的应用场景。探索 Py...

Python 字典合并、求和大作战,轻松搞定各路数据

在使用 Python 采集来自不同渠道的数据,很多都是以字典的形式存储。将它们合并、处理是经常使用的进本操作,在本文中,我们将介绍多种合并和求和字典的方法。合并词典是指涉及将多个词典中的键值对合并到一...

Python 字典(Dictionary):高效数据映射的强大工具

在 Python 中,字典是一种基于键值对(key-value pairs)的数据结构,能够高效地进行数据映射(data mapping)和快速查找(efficient lookup)。字典以无序(u...

Python基础编程——字典的常用方法(三)

前一节介绍了get()、items()、keys()、pop()四种字典的常用方法,本节继续介绍剩余的四种字典常用的方法:popitem()、setdefault()、update()、values(...

Python字典核心秘籍:从键值对到嵌套结构,20+实战案例轻松掌握

Python字典(Dictionary)知识点教程一、字典的定义与特性定义:字典是可变的、无序的键值对集合,用花括号 {} 定义,键值对用 键: 值 表示。person = {"name": "Ali...

Python 中没人告诉你的10个酷炫功能

Python 是一种强调简洁和可读性的语言,但其干净的语法之下隐藏着丰富的强大且不太为人所知的特性。许多开发者,即使是经验丰富的开发者,也常常忽视这些宝石,它们可以帮助简化代码、提高效率,并使编程更加...