我夏天的到来 发表于 2026-4-27 20:28:28

第26讲第4测试题疑问

4. 请目测下边代码执行后,字典dict1的内容是什么?
>>> dict1.fromkeys((1, 2, 3), ('one', 'two', 'three'))
>>> dict1.fromkeys((1, 3), '数字')


课后答案给出得是:
答:执行完成后,字典dict1的内容是:{1: '数字', 3: '数字'}
这里要注意的是,fromkeys方法是直接创建一个新的字典,不要试图使用它来修改一个原有的字典,因为它会直接无情的用把整个字典给覆盖掉。

有几个疑问:
1、dict1根本就没定义,运行不了,会直接报错;
2、假设提前定义dict1={},最后运行之后,只能说:运行结果是{1: '数字', 3: '数字'};而实际上dict1={},我这样理解对吗?
3、既然“fromkeys方法是直接创建一个新的字典,不要试图使用它来修改一个原有的字典”,为什么又说“因为它会直接无情的用把整个字典给覆盖掉”?明明是新创建一个字典,根本就不会改变原有字典,在呢么又“覆盖了”?
我在群里也问过,因为人太多,助理回复之后,我就没有深究了。跑这里来咨询下。看是不是我哪里理解错了。

FishC 发表于 2026-4-27 20:28:43

首先,你的理解完全正确,问题的关键点在于 fromkeys 方法的返回值和使用方式。下面针对你的三个疑问逐一解释:

1. 关于 dict1 未定义
你的观察很准:如果 dict1 从未定义,直接调用 dict1.fromkeys(...) 会引发 NameError,因为 Python 找不到变量 dict1。所以题目给出的代码在真实环境下第一行就会报错,根本不会继续执行第二行。
虽然 fromkeys 是字典的类方法,也可以通过实例调用,但前提是该实例必须存在。

2. 关于 dict1 = {} 的情况
如果你提前定义 dict1 = {},执行两行代码后,dict1 依然是空字典 {}。因为 fromkeys 返回的是 新创建的字典,并不会修改调用它的原始字典对象。所以输出结果只是返回值,而不是修改了 dict1。
正确的操作应该是:
>>> dict1 = {}.fromkeys((1,2,3), ('one','two','three'))
>>> dict1 = dict1.fromkeys((1,3), '数字')
>>> dict1
{1: '数字', 3: '数字'}
但题目代码根本没有赋值,所以 dict1 的内容不变(仍是空或未定义)。

3. 关于“覆盖”和“新建”的矛盾
课后答案中“直接无情的用把整个字典给覆盖掉”这种说法存在误导。实际上,fromkeys 永远不会修改原有字典,它只是创建一个新字典并返回。如果要说“覆盖”,那只有当你把返回值重新赋给原变量时才会发生(例如 dict1 = dict1.fromkeys(...))。题目中并没有赋值操作,所以“覆盖”这个词用在这里是错误的。
正确的理解:

[*] dict.fromkeys(seq, value) 返回一个新字典,原字典不变。
[*] 如果希望通过 fromkeys 更新字典,必须手动赋值: dict1 = dict1.fromkeys(...),这时原字典引用被替换,但 fromkeys 本身依然不修改原对象。


所以,你的疑问很有道理,答案是存在概念性错误的。你之前的理解是正确的:如果 dict1 未定义则报错;如果已定义为空字典,执行后 dict1 还是空字典;fromkeys 只返回新字典,不会覆盖原有的。

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
页: [1]
查看完整版本: 第26讲第4测试题疑问