最近在做项目改代码时凑巧发现一个错误:在for循环中用append()函数遍历进行添加项会造成内容的覆盖。 n条内容,遍历后只剩下最后一个,且重复n遍,自行解决没能完成,通过查阅资料发现了问题(不过也没确保自己弄懂了原理,如有自己见解的可以留言与我讨论) 查阅的网页:https://www.cnpython.com/qa/1410227
解决示例
假设有一个列表l用来添加所需要的内容,要从data这个字典中取出一一对应的内容,让水果中文名与其英文名能对应,实现{‘name’: ‘苹果’, ‘ename’: ‘apple’}这样的效果。 直接放代码:
1.错误情况
def apd():
l = []
d = {}
d['name']=''
d['ename']=''
data = {'name':['苹果','香蕉','梨'],'ename':['apple','banana','pear']}
for i in range(len(data['name'])):
d['name']=data['name'][i]
d['ename']=data['ename'][i]
l.append(d)
print(l)
apd()
运行结果
[{'name': '梨', 'ename': 'pear'}, {'name': '梨', 'ename': 'pear'}, {'name': '梨', 'ename': 'pear'}]
结果取出来的3个字典都是梨的内容。
2.正确代码
更改字典d的定义位置,重新运行代码:
def apd():
l = []
data = {'name':['苹果','香蕉','梨'],'ename':['apple','banana','pear']}
for i in range(len(data['name'])):
d = {}
d['name']=data['name'][i]
d['ename']=data['ename'][i]
l.append(d)
print(l)
apd()
运行结果:
[{'name': '苹果', 'ename': 'apple'}, {'name': '香蕉', 'ename': 'banana'}, {'name': '梨', 'ename': 'pear'}]
发现问题解决,成功取出了所有的水果,并实现中英文名对应。 个人认为是局部变量与全局变量的区别,在for循环内部定义,每次定义的字典d都会分配不一样的地址,从而实现内容的加入。而定义在for循环外部的字典d,占用一个地址,因此,当内容发生改变时,之前存储进去的内容同样发生改变。
3.错误情况的id处理
在for循环中加入id()函数,输出字典d的id:
def apd():
l = []
d = {}
d['name']=''
d['ename']=''
data = {'name':['苹果','香蕉','梨'],'ename':['apple','banana','pear']}
for i in range(len(data['name'])):
d['name']=data['name'][i]
d['ename']=data['ename'][i]
print(id(d))
l.append(d)
print(l)
apd()
运行结果:
2263559432856
2263559432856
2263559432856
[{'name': '梨', 'ename': 'pear'}, {'name': '梨', 'ename': 'pear'}, {'name': '梨', 'ename': 'pear'}]
发现3次的id相同,这是因为字典d定义在了for循环外面。 而如之前的操作更改字典d的位置:
4.正确情况的id处理
def apd():
l = []
data = {'name':['苹果','香蕉','梨'],'ename':['apple','banana','pear']}
for i in range(len(data['name'])):
d = {}
d['name']=data['name'][i]
d['ename']=data['ename'][i]
print(id(d))
l.append(d)
print(l)
apd()
运行结果:
2351310826296
2351310826136
2351310827176
[{'name': '苹果', 'ename': 'apple'}, {'name': '香蕉', 'ename': 'banana'}, {'name': '梨', 'ename': 'pear'}]
可见3次分配的id是不同的,所取得的答案也相同。
小结
也算是在完成项目过程中突然出现的小插曲,加深了对于append()函数使用的印象,如有问题可以与我交流~
|