Python映射函数,通过引用/值传递?


问题内容

我对mapPython中的函数有疑问。

据我了解,该函数不会更改正在操作的列表,而是创建一个新列表并将其返回。这个对吗 ?

另外,我有以下代码

def reflect(p,dir):
  if(dir == 'X'):
    func = lambda (a,b) : (a * -1, b)
   else:
    func = lambda (a,b) : (a, b * -1)
  p = map(func,p)
  print 'got', p

points 是一个元组数组,例如: [(1, 1), (-1, 1), (-1, -1), (1, -1)]

如果我以这种方式调用上述函数:

print points
reflect(points,'X')
print points

列表points不变。在函数内部,打印功能可以正确打印我想要的内容。

有人可以指出我的方向吗,我可以学习所有通过值/引用等传递的内容如何在python中工作,以及如何解决以上问题?或者也许我正在努力尝试在python中模仿Haskell

谢谢

编辑:

说而不是p = map(func,p)

for i in range(len(p)):
  p[i] = func(p[i])

列表的值在函数外部进行更新,就像通过引用进行工作一样。gh,希望这很清楚:S


问题答案:

您会误解引用在Python中是如何工作的。在这里,所有名称都是引用,没有“值”。名称绑定到对象。但是=不会修改名称所指向的对象,而是将名称重新绑定到其他对象:

x = 42
y = x
# now:
# 'is' is a identity operator — it checks whether two names point to the
# exact same object
print x is y # => True
print x, y   # 42 42
y = 69
# now y has been rebound, but that does not change the '42' object, nor rebinds x
print x is y # => False
print x, y   # 42 69

要修改对象本身,它必须是可变的,即,公开对其进行了更改或具有可修改的字典的成员。当您重新绑定时,会发生与上述相同的事情p-它完全没有关系points,它只是修改了本地p名称的含义。

如果要模拟类似C ++的引用,则需要将对象封装到可变容器中,例如列表。

reflect([points], 'X')

# inside reflect:
p[0] = ...

但是,至少在这种情况下,您不应该-应该只返回新对象。

points = reflect(points, 'X')

# inside reflect, instead of p = ...
return map(func, p)

好吧,既然我考虑了一下,您也可以

p[:] = map(func, p)

但是同样,返回新对象通常更好。