1

我一直认为x += 1这只是语法速记(并且完全等同于)x = x + 1,直到我花了一段时间试图弄清楚为什么这段代码没有按预期运行:

[ipython/euler 72 ]$ def func(mylist):
    mylist += random.sample(range(100),2)
    # do stuff with the random result, then restore original list
    mylist = mylist[:-2]

它应该返回它得到的相同列表,但它似乎不是这样工作的:

[ipython/euler 81 ]$ x = [1,2,3]

[ipython/euler 82 ]$ func(x)
[1, 2, 3, 23, 7]

[ipython/euler 83 ]$ func(x)
[1, 2, 3, 23, 7, 42, 36]

[ipython/euler 84 ]$ func(x)
[1, 2, 3, 23, 7, 42, 36, 0, 5]

如果我将赋值语句更改为长格式mylist = mylist + ...,它会按预期工作并保持列表不变。

为什么会这样?我认为这与列表是可变的有关,并且当作为列表的重载方法调用时 iadd 可能不是“真正的”加法,但我仍然希望解释器将它们视为等效的。

4

2 回答 2

4

线

mylist += random.sample(range(100),2)

mylist就地改变列表(这就是为什么它被称为iadd:就地添加)。这意味着它会更改调用者范围内的原始列表。

mylist = mylist[:-2]

创建一个新的本地 对象mylist并将全局的内容分配mylist[:-2]给它。然后在从函数返回时立即丢弃这个新的本地对象。

于 2013-09-19T10:23:03.827 回答
4

如您所知,第一个代码 ( mylist += random.sample(range(100),2)) 修改了列表。

这两个:

mylist = mylist + ...
mylist = mylist[:-2]

创建一个名为 的对象mylist,然后为其赋予与前一个 (或派生自)相同的值mylist

出于这个原因,后一行不会恢复原始列表;它正在创建一个新副本,并单独留下原始参考。

您可以通过更改列表来获得所需的行为:

mylist[:] = mylist[:-2]
于 2013-09-19T10:23:06.880 回答