当前位置: 首页 > 面试题库 >

python上的列表始终附加相同的值[重复]

壤驷康裕
2023-03-14
问题内容

这个问题已经在这里有了答案

为什么这个python生成器每次都返回相同的值? (2个答案)

2年前关闭。

我在while循环中有以下代码。

if gender == 0 and len(men) < 51 :
    height = float((random.uniform(1.3, 1.9) + (random.randint(10, 20)/100.)).__format__('.2f'))
    weight = float((random.uniform(45, 100) * height).__format__('.2f'))
    attr['height'] = height 
    attr['weight'] = weight

    men.append(attr)

因此,此代码始终提供一些随机高度和随机权重。但是异地循环(完成时)。如果这样做print men,我得到以下结果:

[{'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}, {'weight': 76.64, 'height': 1.75}]

它总是一样。不过我的,如果不是使用attr[height] = height; attr['weight] = weight和使用men.append(height); men.append(weight)我得到以下结果:

print men [1.91, 145.95, 1.64, 95.66, 2.0, 159.94, 1.74, 143.36, 1.68, 97.99, 1.6, 90.11, 1.63, 116.2, 1.56, 96.8, 2.04, 198.56, 1.56, 145.96, 1.44, 67.57, 1.83, 94.97, 1.85, 175.69, 1.84, 101.84, 1.54, 135.0, 1.41, 101.23, 1.92, 167.59, 1.74, 142.55, 1.49, 129.07, 1.83, 161.28, 1.59, 97.16, 1.46, 134.53, 2.03, 158.72, 2.05, 184.43, 1.97, 162.81]

如果我在循环内打印attr,它总是具有不同的值(它是我想要的)。但是,当我将其追加到列表中时,列表的值始终相同。我究竟做错了什么?


问题答案:

当前,您的代码的简化示例可以更全面地 说明 结果 为何如此

all_items = []
new_item = {}
for i in range(0,5):
    new_item['a'] = i
    new_item['b'] = i

    all_items.append(new_item)
    print new_item
    print hex(id(new_item))  # print memory address of new_item

print all_items

请注意, 每次 循环 时,
对象的内存地址都相同。这意味着每次添加的对象都是相同的。因此,当您打印最终列表时,您将在循环中的每个位置打印相同对象的坐标。

每次您遍历循环时,值都会被更新-
假设您每天都在同一堵墙上绘画。第一天,可能是蓝色的。第二天,您重新粉刷相同的墙(或对象),然后变成绿色。最后一天,您将其绘制成橙色,然后变为橙色-
同一堵墙现在 总是 橙色。您对attr对象的引用就像说您有同一堵墙。

即使您在涂完油漆后看着墙壁,颜色也会改变。但是之后是橙色的墙-即使您看了5次也是如此。

当我们在每次迭代中将对象设为新对象时,请注意发生了两件事:

  1. 内存地址更改
  2. 这些值作为唯一值保留

这类似于绘制 不同的 墙。在完成最后一幅墙的绘制之后,先前的每面墙仍会以您首先绘制的颜色进行绘制。

您可以在下面的每次迭代中创建每个对象的地方看到它:

all_items = []
for i in range(0,5):
    new_item = {}
    new_item['a'] = i
    new_item['b'] = i

    all_items.append(new_item)
    print hex(id(new_item))


 print all_items

您也可以采用其他方式,例如:

all_items = []
for i in range(0,5):
    new_item = {'a': i, 'b': i}
    all_items.append(new_item)
    print hex(id(new_item))    
print all_items

甚至一步一步:

all_items = []
for i in range(0,5):
    all_items.append({'a': i, 'b': i})

print all_items

因此,以下任何一种都可以工作:

attr = {}
attr['height'] = height 
attr['weight'] = weight

men.append(attr)

要么:

men.append({'height': height, 'weight': weight})


 类似资料:
  • 问题内容: 没有循环,是否有可能将所有列表值初始化为某个布尔值?例如,我要列出所有False的N个元素。 问题答案: 您可以这样做:- 注: - 请注意,你不应该有这样做的有相同的值,否则你会看到令人惊讶的行为像一个在下面的例子: - 如您所见,您在一个内部列表中所做的更改将反映在所有这些内容中。

  • 我试图创建一个包含两个参数的节点的单链表。每当我使用尾指针加入另一个节点时,头指针将取与新节点相同的值。 我确信指针指向相同的内存位置或类似的东西,但我不确定如何修复这个问题。 我想使用这个函数创建一个单链接列表,其中头节点指向列表中的第一个元素,尾节点指向列表中的最后一个元素。当前代码将生成表示最后添加的元素的两个变量。

  • 问题内容: 我打算初始化一个长度为n的列表列表。 但是,这以某种方式将列表链接在一起。 我希望有这样的东西: 有任何想法吗? 问题答案: 问题在于它们在内存中都是完全相同的列表。当使用语法时,得到的是n许多x对象的列表,但是它们都是对同一对象的引用。它们不是不同的实例,而是n对同一实例的引用。 要列出3个不同的列表,请执行以下操作: 这为你提供了3个单独的实例,这是你想要的 类似于 虽然类似于:

  • 问题内容: 有没有更优雅的方式来编写此代码? 我在做什么:我有键和日期。可能有多个日期分配给键,因此我正在创建一个包含日期列表的字典来表示这一点。以下代码可以正常工作,但是我希望有一个更优雅和Pythonic的方法。 我期望下面的工作,但我不断得到一个NoneType没有属性附加错误。 这可能与以下事实有关: 但为什么? 问题答案: 返回,因为它是就地操作,并且您将其分配回。因此,下一次您实际要做

  • Spark-Scala环境中的函数有一些问题。我想在我的DataFrame中添加一个新列,如下所示: 变成: 但是Spark返回给我这个错误: 你能帮我吗?非常感谢!

  • 我正在使用FCM进行消息推送,我正在将数据从通知发送到我正在打开的活动。 建筑通知规范: 在我的活动中,我得到了如下意图: 但问题是,每次Intent Extras在活动中为空时,我都检查了我在社区中找到的所有可能的原因,但每次响应都是相同的。 我试过下面提到的链接 Intent的额外值始终为空 Android通知PendingIntent Extras null 总是从通知意图android获取