在numpy数组副本中交换元素

时间:2018-08-27 19:50:29

标签: python arrays numpy

我有一个numpy数组的列表/数组,表示将对象划分为子组。

我想创建此数组的副本,在其中可以交换子组中的元素,并使原始分组保持不变。

我为此编写的函数是:

def group_swap(groups):
# Chooses two random groups and swaps two random elements from each 
group.
    gr = np.copy(groups)
    g1 = np.random.randint(len(gr))
    g2 = np.random.randint(len(gr))
    if g1 != g2:
        e1 = np.random.randint(len(gr[g1]))
        e2 = np.random.randint(len(gr[g2]))
        gr[g1][e1] ,gr[g2][e2] = gr[g2][e2].copy(),gr[g1][e1].copy()
        return(gr)
    else:
        return(groups)

基于此question,我已经能够交换元素。但是,如本例所示,原始数组中的元素也被交换。

a = np.array_split(np.arange(10),3)
print('orginal before swap: ',a)
a_swap = group_swap(a)
print('original after swap: ',a)
print('swapped array: ',a_swap)

哪个给:

original before swap:  
[array([0, 1, 2, 3]), array([4, 5, 6]), array([7, 8, 9])]
original after swap:  
[array([0, 1, 2, 7]), array([4, 5, 6]), array([3, 8, 9])]
swapped array:  
[array([0, 1, 2, 7]) array([4, 5, 6]) array([3, 8, 9])]

理想情况下,数组a应该保持不变,只有a_swap显示交换的元素。我曾希望在函数中创建并使用数组的副本能够达到目的,但那没有用。

有人可以指出我可能会想念的东西吗?我觉得这是我以后会踢自己的东西。

谢谢

PS:奇怪的是,如果每个组中的元素数量相等,这似乎可以工作,但是我不明白为什么。

   original before swap: 
    [array([0, 1, 2, 3]), array([4, 5, 6, 7]), array([ 8,  9, 10, 11])]
    original after swap:  
    [array([0, 1, 2, 3]), array([4, 5, 6, 7]), array([ 8,  9, 10, 11])]
    swapped array:
    [[ 0  1  8  3]
    [ 4  5  6  7]
    [ 2  9 10 11]]

2 个答案:

答案 0 :(得分:1)

当每个元素中的组件数不相等时,您将获得一个数组列表(嵌套对象)。

当组件数相等时,您将拥有一个二维数组(一个对象)。

您使用的copy称为浅表复制,它仅复制顶层对象(第二种情况下为2d数组,第一种情况下仅复制数组的地址)。因此,在第一种情况下,您的原始数据也会更改。 您应该使用copy模块: https://docs.python.org/3/library/copy.html

答案 1 :(得分:0)

a = np.array_split(np.arange(10),3)
a = np.asarray(a)
b = a.copy() -1 +1
print('orginal before swap: ',a)
a_swap = group_swap(b)
print('original after swap: ',a)
print('swapped array: ',a_swap)

据我所知,ndarray.copy()提取了数组直到的浅表副本。当您在方法中调用变量时,它使用数组的浅表副本,而忽略了要更改的情况。通过在将变量传递给方法之前更改变量,它会将b的内存引用更改为与a不同的位置。

a = np.asarray(a)可以从列表转换为numpy数组,因此-1 +1是有效的操作。 做同一件事的方法可能有很多,这似乎是最简单的。