dict.fromkeys()用同一对象引用作所有键的默认值,可变类型会导致意外同步修改;应使用字典推导式、defaultdict或循环手动构建独立对象。

Python中dict.fromkeys()看似简单,实则暗藏陷阱——它用同一个对象作为所有键的默认值,一旦该值是可变类型(如列表、字典),修改任一键对应的值,其余键也会“意外”同步改变。
fromkeys共享同一对象引用
dict.fromkeys(keys, value)并不会为每个键创建独立副本,而是将value这个对象的引用赋给每一个键。如果value是不可变类型(如0、None、"a"),问题不明显;但若是可变类型,后果立现:
-
错误示范:
d = dict.fromkeys(['a', 'b', 'c'], [])
此时d['a'] is d['b'] is d['c']为True,三者指向同一空列表。 - 执行
d['a'].append(1)后,d变成{'a': [1], 'b': [1], 'c': [1]}——这不是预期行为。
正确创建独立默认值的替代方案
需确保每个键对应一个**新创建的对象**,常用方法有:
-
字典推导式(推荐):
d = {k: [] for k in ['a', 'b', 'c']}—— 每次循环都新建一个空列表。 -
使用
defaultdict:from collections import defaultdictd = defaultdict(list),访问不存在的键时自动调用list()生成新列表。 -
循环手动构建:
d = {}; for k in keys: d[k] = [],明确可控,适合逻辑复杂场景。
为什么None作默认值是安全的?
因为None是不可变单例对象,本身不能被“修改”。即使多个键共用None,后续赋值(如d['a'] = [1])只是让'a'指向新列表,不影响其他键。这和可变对象的就地修改(如.append())有本质区别。
立即学习“Python免费学习笔记(深入)”;
调试与识别该陷阱的小技巧
遇到多键值“联动变化”时,快速验证是否为引用共享:
- 用
id()检查:若id(d['a']) == id(d['b']),说明引用相同。 - 用
is比较:d['a'] is d['b']返回True即为同对象。 - 避免在
fromkeys中传入可变对象,除非你明确需要共享状态(极少见)。










