循环迭代器意外行为
Loop iterator unexpected behavior
我有两个结构非常相似的函数。第一个创建一个 dict
的 3D 列表并添加数据(数据由用于构造列表的三个迭代器的值组成)。第二个仅显示 dict
数据。问题是第二个函数显示的数据是错误的。而不是显示迭代器的值按预期增加:
(0, 9, 19)
(0, 9, 20)
(0, 10, 0)
(0, 10, 1)
(0, 10, 2)
我最终得到显示的迭代器的最大值:
(0, 12, 20)
(0, 12, 20)
(0, 12, 20)
(0, 12, 20)
(0, 12, 20)
代码如下:
xCount = 13
yCount = 21
numPages = 1
skip = [[4, 7], [4, 13], [8, 7], [8, 13]]
def createFrames():
frameData = {'textFrameName': ''}
framesData = [[[frameData for y in range(yCount)] for x in range(xCount)] for p in range(numPages)]
for p in range(numPages):
for x in range(xCount):
for y in range(yCount):
skipFrame = False
for entry in skip:
if entry[0] == x and entry[1] == y:
skipFrame = True
if skipFrame == False:
framesData[p][x][y]['textFrameName'] = str(p) + ', ' + str(x) + ', ' + str(y)
print(framesData[p][x][y]['textFrameName']) # shows p, x, y incrementing
return framesData
def displayData(framesData):
for p in range(numPages):
for x in range(xCount):
for y in range(yCount):
skipFrame = False
for entry in skip:
if entry[0] == x and entry[1] == y:
skipFrame = True
if skipFrame == False:
print(framesData[p][x][y]['textFrameName']) # shows 0, 12, 20 only
# Create text frames.
framesData = createFrames()
# Add data to dictionaries.
displayData(framesData)
使用 python 2.7.16.
问题来自列表理解。您使用可变的 frameData
,因此每次更新它时,您都会更新它。
下面是显示相同行为的简化过程:
>>> a = {"a" : 1}
>>> l = [a for _ in range(3)]
>>> l
[{'a': 1}, {'a': 1}, {'a': 1}]
>>> l[0]["a"] = 2
>>> l
[{'a': 2}, {'a': 2}, {'a': 2}]
您可以使用.copy()
创建一个副本,或者每次在理解中创建对象
>>> a = {"a" : 1}
>>> l2 = [a.copy() for _ in range(3)]
>>> l2[0]["a"] = 2
>>> l2
[{'a': 2}, {'a': 1}, {'a': 1}]
>>> # or
>>> l3 = [{"a" : 1} for _ in range(3)]
>>> l3[0]["a"] = 2
>>> l3
[{'a': 2}, {'a': 1}, {'a': 1}]
我有两个结构非常相似的函数。第一个创建一个 dict
的 3D 列表并添加数据(数据由用于构造列表的三个迭代器的值组成)。第二个仅显示 dict
数据。问题是第二个函数显示的数据是错误的。而不是显示迭代器的值按预期增加:
(0, 9, 19)
(0, 9, 20)
(0, 10, 0)
(0, 10, 1)
(0, 10, 2)
我最终得到显示的迭代器的最大值:
(0, 12, 20)
(0, 12, 20)
(0, 12, 20)
(0, 12, 20)
(0, 12, 20)
代码如下:
xCount = 13
yCount = 21
numPages = 1
skip = [[4, 7], [4, 13], [8, 7], [8, 13]]
def createFrames():
frameData = {'textFrameName': ''}
framesData = [[[frameData for y in range(yCount)] for x in range(xCount)] for p in range(numPages)]
for p in range(numPages):
for x in range(xCount):
for y in range(yCount):
skipFrame = False
for entry in skip:
if entry[0] == x and entry[1] == y:
skipFrame = True
if skipFrame == False:
framesData[p][x][y]['textFrameName'] = str(p) + ', ' + str(x) + ', ' + str(y)
print(framesData[p][x][y]['textFrameName']) # shows p, x, y incrementing
return framesData
def displayData(framesData):
for p in range(numPages):
for x in range(xCount):
for y in range(yCount):
skipFrame = False
for entry in skip:
if entry[0] == x and entry[1] == y:
skipFrame = True
if skipFrame == False:
print(framesData[p][x][y]['textFrameName']) # shows 0, 12, 20 only
# Create text frames.
framesData = createFrames()
# Add data to dictionaries.
displayData(framesData)
使用 python 2.7.16.
问题来自列表理解。您使用可变的 frameData
,因此每次更新它时,您都会更新它。
下面是显示相同行为的简化过程:
>>> a = {"a" : 1}
>>> l = [a for _ in range(3)]
>>> l
[{'a': 1}, {'a': 1}, {'a': 1}]
>>> l[0]["a"] = 2
>>> l
[{'a': 2}, {'a': 2}, {'a': 2}]
您可以使用.copy()
创建一个副本,或者每次在理解中创建对象
>>> a = {"a" : 1}
>>> l2 = [a.copy() for _ in range(3)]
>>> l2[0]["a"] = 2
>>> l2
[{'a': 2}, {'a': 1}, {'a': 1}]
>>> # or
>>> l3 = [{"a" : 1} for _ in range(3)]
>>> l3[0]["a"] = 2
>>> l3
[{'a': 2}, {'a': 1}, {'a': 1}]