【发布时间】:2020-08-12 10:55:48
【问题描述】:
我正在阅读有关通过引用或值复制数组 (&list) 的内容。但是,我在这里遇到了一个问题。为了展示我的问题,我做了三个示例,每个示例都有一个分配和一个更改。
第一个例子:默认是通过引用复制的。
因此,更改会影响 a 和 ArrayA,它们都具有相同的地址。好的
第二个例子: 由于首先计算右侧,*1 不会改变它的值,但会导致按值复制。 (我认为这也可以通过其他几种方式完成,例如使用 copy() 和 ..)
因此,更改只影响 c,具有与 ArrayC 不同的地址。好的
第三个例子: 据我所知,在这里我将 [:] 添加到数组中,从而复制数组(=按值)。可以通过e和ArrayE的不同地址来确认。但是,这种变化不仅影响 e,还影响 ArrayE。对我来说,这几乎是出乎意料的,因为它以前甚至向我显示了不同的地址。为什么?
谢谢你=)
import numpy as np
# Example 1, by reference
ArrayA = np.array([5,2,3,5,4])
ArrayB = np.array( [1,2,3,4])
a = ArrayA
a[1:] += ArrayB
print("{}:\t{},\tid: {}".format("ArrayA",ArrayA, id(ArrayA) ))
print("{}:\t {},\tid: {}".format("ArrayB",ArrayB, id(ArrayB) ))
print("{}:\t{},\tid: {}".format("a",a, id(a) ))
ArrayC = np.array([5,2,3,5,4])
ArrayD = np.array( [1,2,3,4])
# Example 2, by value
c = ArrayC*1
c[1:] += ArrayD
print()
print("{}:\t{},\tid: {}".format("ArrayC",ArrayC, id(ArrayC) ))
print("{}:\t {},\tid: {}".format("ArrayD",ArrayD, id(ArrayD) ))
print("{}:\t{},\tid: {}".format("c",c, id(c) ))
# Example 3, by reference/value?!?!
ArrayE = np.array([5,2,3,5,4])
ArrayF = np.array( [1,2,3,4])
e = ArrayE[:]
e[1:] += ArrayF
print()
print("{}:\t{},\tid: {}".format("ArrayE",ArrayE, id(ArrayE) ))
print("{}:\t {},\tid: {}".format("ArrayF",ArrayF, id(ArrayF) ))
print("{}:\t{},\tid: {}".format("e",e, id(e) ))
ArrayA: [5 3 5 8 8], id: 2450575020480
ArrayB: [1 2 3 4], id: 2450575021680
a: [5 3 5 8 8], id: 2450575020480
ArrayC: [5 2 3 5 4], id: 2450575021280
ArrayD: [1 2 3 4], id: 2450575022080
c: [5 3 5 8 8], id: 2450575022240
ArrayE: [5 3 5 8 8], id: 2450575022640
ArrayF: [1 2 3 4], id: 2450575022000
e: [5 3 5 8 8], id: 2450575022880
【问题讨论】:
-
“通过引用复制”不是一回事。您应该阅读nedbatchelder.com/text/names.html,然后重新提出您的问题。
-
它不是“引用复制”,它根本不是复制。请注意,这与传递引用和传递值无关,它们是评估策略,即函数的参数如何/何时的语义评估。请注意,Python 既不使用按值调用也不使用按引用调用。
-
当您切片
numpy.ndarray对象时,它会创建一个新的 array object,它是数组中底层缓冲区的 view切片。
标签: python arrays list pass-by-reference pass-by-value