Python 字典键 “三变一” 之谜

news2025/12/16 16:34:11
开头:读者的“玄学”字典谜题

上周,朋友发来了一段让他抓耳挠腮的代码:

>>> {True: 'foo', 1: 'bar', 1.0: 'baz'}  
{True: 'baz'}  

“我明明定义了布尔True、整数1、浮点数1.0三个键,结果字典里只剩True一个键,值还变成了最后一个'baz'!这是啥情况?”

这条消息让我想起了当年自己初学 Python 时踩过的类似坑 —— 看似 “不同” 的键,在字典里却被 “合并” 了。今天,咱们就用这个新例子当 “线索”,一起拆解 Python 字典的底层逻辑。

第一步:字典的“盖楼”规则

要搞懂三个键为何只剩一个,得先明白字典是怎么“盖楼”的。

简单来说,字典的构建像搭乐高:先拼一个空架子(空字典),再按顺序往架子上装“键值对模块”。上面的代码等价于:

# 1. 拼空架子  
my_dict = {}  
# 2. 装第一个模块:键是True,值是'foo'  
my_dict[True] = 'foo'  
# 3. 装第二个模块:键是1,值是'bar'  
my_dict[1] = 'bar'  
# 4. 装第三个模块:键是1.0,值是'baz'  
my_dict[1.0] = 'baz'  

重点来了:字典的键是“喜新厌旧”的——如果后装的键和已存在的键“本质相同”,就会覆盖旧值。但问题是:True(布尔)、1(整数)、1.0(浮点数)明明是三种不同的类型,怎么就“本质相同”了?

第二步:True是“伪装的1”

要破解“键相同”的谜题,得从Python的类型关系说起。

在Python的世界里,布尔(bool)是整数(int)的“亲儿子”——官方文档明确写着:

“布尔类型是整数类型的子类型,True等价于整数1False等价于整数0。在大多数上下文中,布尔值的行为与对应的整数值一致。”

这意味着:

  • True == 1 → 是真的(True
  • 1 == 1.0 → 也是真的(浮点数1.0的数值等于整数1
  • 所以True == 1 == 1.0 → 全等于!

用代码验证:

>>> True == 1  
True  
>>> 1 == 1.0  
True  
>>> True == 1 == 1.0  
True  

原来,在字典的“视角”里,这三个键根本就是“同一个人”!所以当依次插入True: 'foo'1: 'bar'1.0: 'baz'时,后两次插入都是在“修改同一个键的值”,最终只保留最后一次的'baz'

第三步:哈希值——字典的“身份证号”

但这里还有个疑问:就算三个键“数值相等”,字典怎么确定它们是“同一个键”?难道只看==吗?

这就要说到字典的底层“黑科技”——哈希表(Hash Table)。字典能快速查找键值对,全靠哈希值:每个键会先通过__hash__方法生成一个哈希值(类似“身份证号”),字典根据这个号码把键“扔”到对应的“抽屉”里;查找时,也先算哈希值,再去对应的抽屉里找。

关键规则是:只有当两个键的哈希值相同,且==返回True时,字典才会认为它们是同一个键

验证这三个键的哈希值:

>>> hash(True)  
1  
>>> hash(1)  
1  
>>> hash(1.0)  
1  

三个键的哈希值都是1==又全返回True,字典自然把它们当同一个键。所以后插入的1: 'bar'1.0: 'baz',本质上都是在修改True对应的值。

第四步:为什么键是True而不是11.0

最后一个疑问:三个键数值相等、哈希相同,为什么最终字典的键是True,而不是后插入的11.0

这涉及字典的“键保留规则”:当多个键被视为相同时,字典会保留第一个插入的键对象。比如:

>>> temp = {1.0: 'test'}  
>>> temp[True] = 'update'  
>>> temp  
{1.0: 'update'}  

这里先插入1.0,后插入True(与1.0相等),字典会保留第一个键1.0,并更新它的值。回到原问题,原字典第一个插入的键是True,所以最终键是True,值被后续插入的'bar''baz'覆盖。

结论:三个“不同”键的终极真相

现在,我们可以彻底解开这个“变脸字典”的谜题了:

  1. 类型关系是根源:Python中boolint的子类,True等价于11又等价于1.0(数值相等)。
  2. 哈希值是身份证:三个键的哈希值都是1,字典通过“哈希值相同+==为True”判定它们是同一个键。
  3. 先到先得保键形:字典保留第一个插入的键对象(True),后续插入只更新值,不修改键。

所以,最终结果{True: 'baz'}的本质是:三个键被字典视为同一对象,后插入的值覆盖了前值,而键保留了第一个插入的True

写在最后:这行代码教会我的事

这个看似“玄学”的字典表达式,其实藏着Python最核心的设计逻辑:

  • 布尔类型的“隐藏身份”(int子类)
  • 字典的哈希表底层逻辑(哈希值+相等性双重校验)
  • 键值对的插入顺序对结果的影响

下次遇到类似“反直觉”的代码时,别急着怀疑语言bug——打开Python解释器,用==hash()验证一下,你会发现Python的底层逻辑远比想象中严谨。

毕竟,Python的“奇怪”,往往藏着最精妙的设计。

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

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

相关文章

element-ui分页的使用及修改样式

1.安装 npm install element-ui -S 2.在main.js中引入,这里是全部引入,也可以按需引入 import ElementUI from element-ui import element-ui/lib/theme-chalk/index.css Vue.use(ElementUI) 3.使用 layout"prev, pager, next, jumper" :jumpe…

从数据中台到数据飞轮:数字化转型的演进之路

从数据中台到数据飞轮:数字化转型的演进之路 数据中台 数据中台是企业为整合内部和外部数据资源而构建的中介层,实现数据的统一管理、共享和高效利用,目标是打破信息孤岛,提高数据使用效率,支持业务决策和创新 实施成本…

2025年5月-信息系统项目管理师高级-软考高项一般计算题

决策树和期望货币值 加权算法 自制和外购分析 沟通渠道 三点估算PERT 当其他条件一样时,npv越大越好

zst-2001 上午题-历年真题 算法(5个内容)

回溯 算法 - 第1题 找合适的位置,如果没有位置就按B回家 d 分治 算法 - 第2题 b 算法 - 第3题 a 算法 - 第4题 划分一般就是分治 a 算法 - 第5题 分治 a 0-1背包 算法 - 第6题 c 算法 - 第7题 最小的为c 3100 c 算法 - 第8题 …

udp多点通信和心跳包

刷题 # UDP多点通信核心要点## 基础通信模式### 单播通信- 一对一通信方式- UDP默认通信模式- 地址指向具体目标主机### 广播通信- 一对多通信机制- 地址范围:xxx.xxx.xxx.255- 仅限局域网传输- 需设置SO_BROADCAST标志### 组播通信- 多对多群组通信- 地址范围&…

音视频学习:使用NDK编译FFmpeg动态库

1. 环境 1.1 基础配置 NDK 22b (r22b)FFmpeg 4.4Ubuntu 22.04 1.2 下载ffmpeg 官网提供了 .tar.xz 包,可以直接下载解压: wget https://ffmpeg.org/releases/ffmpeg-4.4.tar.xz tar -xvf ffmpeg-4.4.tar.xz cd ffmpeg-4.41.3 安装基础工具链 sudo …

如何使用 Qwen3 实现 Agentic RAG?

今天,我们将学习如何部署由阿里巴巴最新Qwen 3驱动的Agentic RAG。 这里是我们的工具栈: CrewAI用于代理编排。 Firecrawl用于网络搜索。 LightningAI的LitServe用于部署。 顶部的视频展示了这一过程。 图表显示了我们的Agentic RAG流程&#xff1…

相机、雷达标定工具,以及雷达自动标定的思路

本篇我们来看一下自动驾驶传感器配置一个非常重要的模块,也就是传感器的标定。这里主要是对我之前修改的功能包的使用进行一个介绍. 对应的资源也已经上传了,0积分下载 安装 首先整个项目是使用ros1来进行启动的,但是要想正常编译,需要先安装三个对应的…

vsomeip环境搭建保姆级教程

vsomeip环境搭建保姆级教程 ubuntu环境搭建 {% links %} site: VMware搭建ubuntu保姆级教程 url: https://zhuanlan.zhihu.com/p/1903219373906327339 desc: flechazo image: https://q1.qlogo.cn/g?b=qq&nk=2861099&s=5 color: “#9d5b8b” {% endlinks %} vsomei…

我的MCP相关配置记录

1.VSCode的Cline中的MCP {"mcpServers": {"github.com/modelcontextprotocol/servers/tree/main/src/github": {"autoApprove": [],"disabled": false,"timeout": 60,"command": "cmd","args&quo…

我们来学nacos -- 集群nacos2.5.1mysql8.4

2.5.1集群搭建 架构下载解压到3个文件夹初始化数据库&数据迁移检查端口可用配置cluster.confapplication.properties 使用mysql8.4的jar启动db.num is null报错datasource错误成功 nginx反向代理集群查看 架构 其中包含3个nacos节点,然后一个负载均衡器代理3个…

Rollup入门与进阶:为现代Web应用构建超小的打包文件

我们常常面临Webpack复杂配置或是Babel转译后的冗余代码,结果导致最终的包体积居高不下加载速度也变得异常缓慢,而在众多打包工具中Rollup作为一个轻量且高效的选择,正悄然改变着这一切,本文将带你深入了解这个令人惊艳的打包工具…

专题四:综合练习( 找出所有子集的异或总和再求和)

以leetcode1863题为例 题目分析: 找到每个子集,然后子集中的元素异或之后全部相加 算法原理分析: 画决策树:第一层为这个子集有一个元素 第二层这个子集有两个元素 从上往下罗列,把所有子集都罗列出来&#xf…

STM32 修炼手册

第一章 计算机体系结构(了解) 后续在板子上开发的时候,需要考虑是否有操作系统 方式一:有操作系统,通过c库通过os api操作硬件方式二:无操作系统, 通过c库通过固件库操作硬件 第二章 STM32开发板概述 板子/开发板&…

缓存(2):数据一致性

概述 一致性就是数据保持一致,在分布式系统中,可以理解为多个节点中数据的值是一致的。 强一致性:这种一致性级别是最符合用户直觉的,它要求系统写入什么,读出来的也会是什么,用户体验好,但实现起来往往对系统的性能影响大弱一致性:这种一致性级别约束了系统在写入成功…

ppy/osu构建

下载 .NET (Linux、macOS 和 Windows) | .NET dotnet还行 构建:f5 运行:dotnet run --project osu.Desktop -c Debug

基于几何布朗运动的股价预测模型构建与分析

基于几何布朗运动的股价预测模型构建与分析 摘要 本文建立基于几何布朗运动的股价预测模型,结合极大似然估计与蒙特卡洛模拟,推导股价条件概率密度函数并构建动态预测区间。实证分析显示模型在标普500指数预测中取得89%的覆盖概率,波动率估…

python如何提取Chrome中的保存的网站登录用户名密码?

很多浏览器都贴心地提供了保存用户密码功能,用户一旦开启,就不需要每次都输入用户名、密码,非常方便。作为python脚本,能否拿到用户提前保存在浏览器中的用户名密码,用以自动登录呢?必须有,小爬…

Redis实现分布式获取全局唯一自增ID的案例。

【1】简易自增版本(从 1 开始 1,2,3,...) 项目结构 下面是一个基于 RedisTemplate 实现的分布式全局唯一自增 ID 生成器的案例。适用于 Java Spring Boot 环境,利用 Redis 的原子操作 INCR 指令。 ✅ 原理说明 Redis 提供的 INCR 命令是原子性的&…

人脸识别备案:筑牢人脸信息 “安全墙”

人脸识别备案制度主要依据《人脸识别技术应用安全管理办法》建立,人脸识别技术广泛应用于安防、金融、门禁、交通等领域,带来便利高效的同时,人脸信息安全问题也引发担忧。为规范技术应用、保护个人信息权益,人脸识别备案制度应运…