本文解决python中比较令人困惑的一个小问题:传递到函数中的参数若在函数中进行了重新赋值,对于函数外的原变量有何影响。看一个小栗子:

def fun(a):
a=2
return
a=1
fun(a)

请问程序执行后,a=1还是2?并解释原因。

实际上,这个问题有两个比较tricky的地方:

1. python的变量是没有类型的,类型属于对象。也就是说当我们操作x=6的时候,6是一个int类型的对象,而x就是个名字,其指针指向6这个对象。除此之外,x可以指向任何类型的对象,哪怕先后指向不同类型的对象也不会出错。

Mutable and Immutable Variables in Python-LMLPHP

2. python中的对象分为mutable和immutable两种,二者在作为参数传递时有根本的区别。各个类型的对象分类见下表:

Mutable and Immutable Variables in Python-LMLPHP

首先,代码中a=1的意思是,创建变量a(指针a),指向数字1这个对象的地址。在调用fun(a)的时候,因为数字对象是immutable的,所以传递到fun函数中的参数,实际上并不是a,而是a的复制品,暂且说成是b。b也是指向1这个对象的,所以说,之后在函数内部无论再怎么修改这个参数,和之前的变量a已经没有关系了。所以程序的执行结果是1。

3. Accessor and Mutator

在面向对象语言中,一个对象中有可能存在两种方法:Accessor method以及Mutator.

例如:

Mutable and Immutable Variables in Python-LMLPHP

str的upper()方法就是一个Accessor Method,因为它并不会改变x的值。再看一个Mutator的例子:

Mutable and Immutable Variables in Python-LMLPHP

调用reverse()方法后,myList中的顺序被改变了,说明reverse()是一个Mutator方法。所有的对象都有Accessor方法,但并不是都有Mutator方法。具有Mutator方法的对象是Mutable的,仅有Accessor的对象是Immutable的。

05-25 15:08