dict - python使用{}.fromkeys创建字典后,通过append更新list型value出错?
天蓬老师
天蓬老师 2017-04-17 15:54:46
[Python讨论组]

直接上代码:
header= {'a', 'c', 'b', 'e', 'f'}
然后根据该序列初始化字典dic1,并设置其值默认格式为list:
dic1={}.fromkeys(header,[])
print dic1
输出:{'a': [], 'c': [], 'b': [], 'e': [], 'f': []}
一切都是对的:

然后为key为'a'的键更新value:
dic1['a'].append(1)
print dic1
可输出竟然是:
{'a': [1], 'c': [1], 'b': [1], 'e': [1], 'f': [1]}

按道理应该是
{'a': [1], 'c': [], 'b': [], 'e': [], 'f': []}

可是换一种方式:
直接定义字典的话:
dic2={'a': [], 'c': [], 'b': [], 'e': [], 'f': []}
dic2['a'].append(1)
print dic2
输出则是正常的
{'a': [1], 'c': [], 'b': [], 'e': [], 'f': []}
在pydev中对比,两个变量是一模一样的:

这是为什么呢,求大神支招。

天蓬老师
天蓬老师

欢迎选择我的课程,让我们一起见证您的进步~~

全部回复(4)
PHP中文网

楼上的解决方案是对的,不过Python2.7及以上版本才支持这么写,如果你的版本不支持,用这个:

>>> dic1 = dict([(k, []) for k in header])
>>> dic1
{'a': [], 'c': [], 'b': [], 'e': [], 'f': []}
>>> map(id, dic1.values())
[38590424, 38590024, 38591464, 38757120, 31261600]
大家讲道理

原因是第一种情况的列表都是一个对象,第二种则都是不同的对象。他们的key的变量地址都是相同

In [1]: header= {'a', 'c', 'b', 'e', 'f'}

In [2]: dic1={}.fromkeys(header,[])

In [3]: dic1
Out[3]: {'a': [], 'b': [], 'c': [], 'e': [], 'f': []}

In [4]: dic1['a'] is dic1['b']
Out[4]: True

In [5]: dic2={'a': [], 'c': [], 'b': [], 'e': [], 'f': []}

In [6]: dic2['a'] is dic2['b']
Out[6]: False

In [8]: dic1.keys()
Out[8]: ['a', 'c', 'b', 'e', 'f']

In [9]: dic1.keys()[0] is dic2.keys()[0]
Out[9]: True
伊谢尔伦

Python 中 list 是可变类型,在作为参数传递的时候是传引用的。

所以 fromkeys 函数把所有 key 都指向了同一个空列表。改动其中任何一个,其他的都改掉了。

高洛峰

前面都给的分析, 我上个代码:

dic1 = {k:[] for k in header}  # fromkeys改成这个

改动这一行就行了

热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号