鱼C论坛

 找回密码
 立即注册
查看: 1843|回复: 7

[已解决]这是怎么回事?字典写入与预期不符。

[复制链接]
发表于 2022-5-28 20:21:24 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
现象如下:
a = [1,2,3,4,5,6,7,8,9,10]
b = ['a', 'b', 'c']
c = dict.fromkeys(b, [])

for i in range(3):
    for k in c.keys():
        c[k].append(a.pop())

        
c
{'a': [10, 9, 8, 7, 6, 5, 4, 3, 2], 'b': [10, 9, 8, 7, 6, 5, 4, 3, 2], 'c': [10, 9, 8, 7, 6, 5, 4, 3, 2]}

执行完的预期:
{‘a': [10, 7, 4], 'b': [9, 6, 3], 'c': [8, 5, 2]}
最佳答案
2022-5-29 10:37:08
傻眼貓咪 发表于 2022-5-28 22:35
一般不建议在循环里用 pop() 或 removed(),因为一旦你 pop() 了元素,下标就改变了,而循环只会继续,比 ...



你所说的内容和楼主遇到的问题没关系,这里出问题的主要原因如下:

执行下面这个代码就知道为什么:
a = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
b = ['a', 'b', 'c']
c = dict.fromkeys(b, [])

print(id(c['a']))
print(id(c['b']))
print(id(c['c']))

这里代码会打印同一个地址,说明三个键所指向的列表为同一个列表

formkeys 方法创建的字典,字典的键所指向的值都是一个列表,即一个地址的,这也是楼主代码出错的原因

这里导致了你 append 添加的都是同一个列表,所以打印出字典后三个键指向同一个列表的情况

所以不可变数据类型可以用 formkeys 进行初始化,而不推荐初始化为可变数据类型时候使用 formkeys

楼主代码可以这样实现:
a = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
b = ['a', 'b', 'c']
c = {i:[] for i in b}

for i in range(3):
    for k in c.keys():
        c[k].append(a.pop())
        
print(c)
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2022-5-28 21:04:54 | 显示全部楼层
a = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
b = ['a', 'b', 'c']
c = dict.fromkeys(b, None)

for i, key in enumerate(b[::-1], start = 1):
    c[key] = [num for num in a[i::3] if num != 1]

print(c)
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2022-5-28 21:21:38 | 显示全部楼层
a = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
b = ['a', 'b', 'c']
c = dict.fromkeys(b, None)
c
{'a': None, 'b': None, 'c': None}

for i, key in enumerate(b[::-1], start=1):
                        c[key] = [num for num in a[i::3] if num != 1]

                        
c                  
{'a': [4, 7, 10], 'b': [3, 6, 9], 'c': [2, 5, 8]}

实现跟预期还是不符,结果c要求键是a/b/c;值是a.pop(),倒序存入字典c中。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2022-5-28 22:06:51 | 显示全部楼层
allen-xy 发表于 2022-5-28 21:21
实现跟预期还是不符,结果c要求键是a/b/c;值是a.pop(),倒序存入字典c中。

这样??
a = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
b = ['a', 'b', 'c']
c = dict.fromkeys(b, None)

for i, key in enumerate(b[::-1], start = 1):
    c[key] = [num for num in a[i::3] if num != 1]
    c[key].reverse()

print(c)
{'a': [10, 7, 4], 'b': [9, 6, 3], 'c': [8, 5, 2]}
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2022-5-28 22:22:58 | 显示全部楼层


感谢你的实现。
我的问题是:为什么我的代码实现不了我的预期?是哪里逻辑出问题了?
感觉第2个for语句不是依次执行,而是一次性把单个值赋给三个键。
或者说是pop()出来的单个元素被依次赋给了三个键。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2022-5-28 22:35:39 | 显示全部楼层
allen-xy 发表于 2022-5-28 22:22
感谢你的实现。
我的问题是:为什么我的代码实现不了我的预期?是哪里逻辑出问题了?
感觉第2个for语 ...

一般不建议在循环里用 pop() 或 removed(),因为一旦你 pop() 了元素,下标就改变了,而循环只会继续,比如:
arr = [1, 2, 3, 4, 5]

for i in arr:
    print(arr.pop())
5
4
3

只打印 5、4、3 三个数字而已,因为每次 pop() 下标就不一样了。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2022-5-29 10:37:08 | 显示全部楼层    本楼为最佳答案   
傻眼貓咪 发表于 2022-5-28 22:35
一般不建议在循环里用 pop() 或 removed(),因为一旦你 pop() 了元素,下标就改变了,而循环只会继续,比 ...



你所说的内容和楼主遇到的问题没关系,这里出问题的主要原因如下:

执行下面这个代码就知道为什么:
a = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
b = ['a', 'b', 'c']
c = dict.fromkeys(b, [])

print(id(c['a']))
print(id(c['b']))
print(id(c['c']))

这里代码会打印同一个地址,说明三个键所指向的列表为同一个列表

formkeys 方法创建的字典,字典的键所指向的值都是一个列表,即一个地址的,这也是楼主代码出错的原因

这里导致了你 append 添加的都是同一个列表,所以打印出字典后三个键指向同一个列表的情况

所以不可变数据类型可以用 formkeys 进行初始化,而不推荐初始化为可变数据类型时候使用 formkeys

楼主代码可以这样实现:
a = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
b = ['a', 'b', 'c']
c = {i:[] for i in b}

for i in range(3):
    for k in c.keys():
        c[k].append(a.pop())
        
print(c)
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 2 反对 0

使用道具 举报

发表于 2022-5-29 11:44:06 | 显示全部楼层
Twilight6 发表于 2022-5-29 10:37
你所说的内容和楼主遇到的问题没关系,这里出问题的主要原因如下:

执行下面这个代码就知道为什么 ...

好的
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-1-11 12:58

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表