Python 函数中,参数是传值,仍是传引用?

在 C/C++ 中,传值和传引用是函数参数传递的两种方式,在Python中参数是如何传递的?回答这个问题前,不如先来看两段代码。html

 

代码段1:python

def foo(arg):

    arg = 2

    print(arg)

 

a = 1

foo(a)  # 输出:2

print(a) # 输出:1

看了代码段1的同窗可能会说参数是值传递。面试

 

代码段2:app

def bar(args):

    args.append(1)

 

b = []

print(b)# 输出:[]

print(id(b)) # 输出:4324106952

bar(b)

print(b) # 输出:[1]

print(id(b))  # 输出:4324106952

看了代码段2,这时可能又有人会说,参数是传引用,那么问题来了,参数传递究竟是传值仍是传引用或者二者都不是?为了把这个问题弄清楚,先了解 Python 中变量与对象之间的关系。函数

 

变量与对象spa

 

Python 中一切皆为对象,数字是对象,列表是对象,函数也是对象,任何东西都是对象。而变量是对象的一个引用(又称为名字或者标签),对象的操做都是经过引用来完成的。例如,[]是一个空列表对象,变量 a 是该对象的一个引用.net

a = []

a.append(1)

在 Python 中,「变量」更准确叫法是「名字」,赋值操做 = 就是把一个名字绑定到一个对象上。就像给对象添加一个标签。code

a = 1

 

整数 1 赋值给变量 a 就至关因而在整数1上绑定了一个 a 标签。htm

a = 2

 

整数 2 赋值给变量 a,至关于把原来整数 1 身上的 a 标签撕掉,贴到整数 2 身上。对象

b = a

 

把变量 a 赋值给另一个变量 b,至关于在对象 2 上贴了 a,b 两个标签,经过这两个变量均可以对对象 2 进行操做。

 

变量自己没有类型信息,类型信息存储在对象中,这和C/C++中的变量有很是大的出入(C中的变量是一段内存区域)

 

函数参数

 

Python 函数中,参数的传递本质上是一种赋值操做,而赋值操做是一种名字到对象的绑定过程,清楚了赋值和参数传递的本质以后,如今再来分析前面两段代码。

def foo(arg):

    arg = 2

    print(arg)

 

a = 1

foo(a)  # 输出:2

print(a) # 输出:1

 

在代码段1中,变量 a 绑定了 1,调用函数 foo(a) 时,至关于给参数 arg 赋值 arg=1,这时两个变量都绑定了 1。在函数里面 arg 从新赋值为 2 以后,至关于把 1 上的 arg 标签撕掉,贴到 2 身上,而 1 上的另一个标签 a 一直存在。所以 print(a) 仍是 1。

 

再来看一下代码段2

def bar(args):

    args.append(1)

 

b = []

print(b)# 输出:[]

print(id(b)) # 输出:4324106952

bar(b)

print(b) # 输出:[1]

print(id(b))  # 输出:4324106952

 

执行 append 方法前 b 和 arg 都指向(绑定)同一个对象,执行 append 方法时,并无从新赋值操做,也就没有新的绑定过程,append 方法只是对列表对象插入一个元素,对象仍是那个对象,只是对象里面的内容变了。由于 b 和 arg 都是绑定在同一个对象上,执行 b.append 或者 arg.append 方法本质上都是对同一个对象进行操做,所以 b 的内容在调用函数后发生了变化(但id没有变,仍是原来那个对象)

 

最后,回到问题自己,到底是是传值仍是传引用呢?说传值或者传引用都不许确。非要安一个确切的叫法的话,叫传对象(call by object)。若是做为面试官,非要考察候选人对 Python 函数参数传递掌握与否,与其讨论字面上的意思,还不如来点实际代码。

 

show me the code

 

def bad_append(new_item, a_list=[]):

    a_list.append(new_item)

    return a_list

 

这段代码是初学者最容易犯的错误,用可变(mutable)对象做为参数的默认值。函数定义好以后,默认参数 a_list 就会指向(绑定)到一个空列表对象,每次调用函数时,都是对同一个对象进行 append 操做。所以这样写就会有潜在的bug,一样的调用方式返回了不同的结果。

>>> print bad_append('one')

['one']

>>> print bad_append('one')

['one', 'one']

 

 

而正确的方式是,把参数默认值指定为None

def good_append(new_item, a_list=None):

    if a_list is None:

        a_list = []

    a_list.append(new_item)

    return a_list

  

参考:http://python.net/~goodger/projects/pycon/2007/idiomatic/handout.html#other-languages-have-variables

相关文章
相关标签/搜索