问题描述
在Raspbian Pi的Raspbian上的Spyder 3.1.3中使用Python3.5.3.将两个numpy数组附加到名为"list0"的列表中,可以很好地与分配的numpy数组"a"一起使用,例如:
Using Python3.5.3 in Spyder 3.1.3 on Raspbian on a Raspberry Pi.Appending two numpy-arrays to a list named 'list0' works well with allocated numpy arrays 'a' like:
import numpy as np
list0 = []
a = np.array([[1,2,3],[2,3,4]])
list0.append(a)
a = np.array([[11,12,13],[12,13,14]])
list0.append(a)
print("list0 =",list0)
效果很好,可以作为输出(为帖子提供更好的格式):
works well, giving as output (a bit better formatted for the post):
list0 = [ array([[ 1, 2, 3], [ 2, 3, 4]]),
array([[11, 12, 13], [12, 13, 14]]) ]
使用循环将分配替换为a,会发生奇怪的事情:
Replacing the assignment to a using a loop, weird things happen:
import numpy as np
a = np.empty((3), int)
list0 = []
for idx in range(4):
for i in range(3):
a[i] = idx*10 + i
print("idx =",idx,"; a =",a)
list0.append(a)
print("list0 =",list0)
第二行告诉Python使用的数组的形状(在我的原始情况下,它是三维数组).为了进行验证,将打印出名为"a"的生成数组.最后,将新填充的数组'a'添加到'list0'会显示最后一行的四倍.
The second line tells Python the shape of the array used (in my original case it is a three-dimensional array). For verification the generated arrays named 'a' are printed out. Appending the newly filled arrays 'a' to 'list0' finally shows four times the last line.
idx = 0 ; a = [ 0 1 2]
idx = 1 ; a = [10 11 12]
idx = 2 ; a = [20 21 22]
idx = 3 ; a = [30 31 32]
list0 = [ array([30, 31, 32]), array([30, 31, 32]),
array([30, 31, 32]), array([30, 31, 32]) ]
我想'list0'只是包含指向数组'a'的指针的四倍,而该数组'a'仅存在于一个实例/内存范围内.
I suppose that 'list0' simply contains four times a pointer to the array 'a' which only exists in one instance / memory range.
所以:我该如何物理地将每个不同的数组"a"追加(复制?)到列表中?它是python的bug还是仅仅是我对某些东西的误解?当然,我应该考虑更多的pythonian; c)
So: How can I physically append (copy?) each of the different arrays 'a' to the list? Is it a python bug or is it simply my misunderstanding of something? Certainly I should think more pythonian ;c)
感谢您的帮助,彼得
推荐答案
问题
您要将相同的数组 a
附加到您的 list0
4次.像 a
这样的数组是可变对象,这意味着,除其他外,当您为它们分配值时,基础对象也会更改.由于阵列在您的列表中出现了4次,因此这些更改(似乎)显示在4个不同的位置.
The problem
You're appending the same array a
to your list0
4 times. Arrays like a
are mutable objects, which means, among other things, that when you assign values to them the underlying object changes. Since the array is present in your list 4 times, those changes (seem to) show up in 4 different places.
只需少量更改,即可修复您拥有的代码.将数组的副本而不是数组本身附加到列表中:
You can fix the code you have with one small change. Append a copy of the array to your list, instead of the array itself:
import numpy as np
a = np.empty((3), int)
list0 = []
for idx in range(4):
for i in range(3):
a[i] = idx*10 + i
print("idx =",idx,"; a =",a)
list0.append(a.copy())
print("list0 =",list0)
输出:
idx = 0 ; a = [0 1 2]
idx = 1 ; a = [10 11 12]
idx = 2 ; a = [20 21 22]
idx = 3 ; a = [30 31 32]
list0 = [array([0, 1, 2]), array([10, 11, 12]), array([20, 21, 22]), array([30, 31, 32])]
优化解决方案
Python/Numpy提供了许多更好的方法(就使用更少的代码行和更快地运行而言)来初始化数组.对于这样的范围,这是一种合理的方法:
Optimized solution
Python/Numpy offer many better ways (both in terms of using fewer lines of code and running faster) to initialize arrays. For a bunch of ranges like this, here is a reasonable approach:
list0 = [np.arange(n*10, n*10+3) for n in range(4)]
print(list0)
输出:
[array([0, 1, 2]), array([10, 11, 12]), array([20, 21, 22]), array([30, 31, 32])]
您可能还考虑仅使用单个2D数组来代替数组列表.一个单一的数组通常比列表中数组的异构混合更容易使用.这是您的操作方式:
You might also consider just using a single 2D array in place of a list of arrays. One single array is often easier to work with than a heterogenous mix of arrays in a list. Here's how you do that:
arr0 = np.array([np.arange(n*10, n*10+3) for n in range(4)])
print(arr0)
输出:
[[ 0 1 2]
[10 11 12]
[20 21 22]
[30 31 32]]
这篇关于将numpy数组附加到列表-奇怪的事情的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持!