问:
参数是按引用传递还是按值传递? 我如何通过引用传递,以便下面的代码输出 'Changed' 而不是 'Original'?
代码语言:javascript复制class PassByReference:
def __init__(self):
self.variable = 'Original'
self.change(self.variable)
print(self.variable)
def change(self, var):
var = 'Changed'
test = PassByReference()
答:
这个问题源于对Python中变量的误解。如果你习惯了大多数传统语言,你会有一个以下序列语句会发生什么的思维模型:
代码语言:javascript复制a = 1
a = 2
你认为a是一个存储值1的内存位置,然后它被更新为存储值2。这不是Python的工作方式。相反,a一开始是对值为1的对象的引用,由于整数是不可变对象,第二次赋值不是去改变整数对象1 的值,而是创建一个新的整数对象(值为 2)并将其赋给 a。即使a不再引用第一个对象,这两个对象也可能继续共存;事实上,它们可以被程序内的任何数量的其他引用共享。
记住,在Python中,实参是通过赋值方式传递的。由于赋值操作只是创建对象的引用,因此调用者和被调用者中的参数名之间没有别名,本质上也就不存在按引用调用的方式。
实现提问者需求的变通方法是传递一个可变对象。因为两个引用都引用同一个对象,所以对对象的任何更改都会反映在两个位置。
代码语言:javascript复制class PassByReference:
def __init__(self):
self.variable = ['Original']
self.change(self.variable)
print(self.variable)
def change(self, var):
var[0] = 'Changed'
test = PassByReference()
参考:
- stackoverflow question 986006
- https://docs.python.org/3/faq/programming.html#why-did-changing-list-y-also-change-list-x
- https://docs.python.org/3/faq/programming.html#how-do-i-write-a-function-with-output-parameters-call-by-reference