最近学习用Python编程,但是遇到用函数处理数据的时候,对于数据的传递形式优点理解不清,以至于无法用Python使用链表等数据结构。在此特地记录几点学习经验。

总结

传值、引用这个是c/c++、java中的概念,Python中一切都是对象,实参向形参传递的是对象的引用值。就像Python赋值的意思。

Python函数传递的是对象的引用值,非传值或传引用:

如果对象是不可变的,感觉和c语言中传值差不多。

如果对象是可变的,感觉和c语言中传引用差不多。

示例

1. 不可变对象 e.g.元组

"""Python"""

def foo(a):

a = a + (333, 444) #元组不可变,此处对元组进行连接组合

return a

def test():

t = (1, 2, 3)

print foo(t) #打印函数返回的元组

print t

运行test,得到如下结果:

(1, 2, 3, 333, 444)

(1, 2, 3)

元组t 未被函数改变

2. 可变对象 e.g.列表

"""Python"""

def foo(a):

"""在函数内部直接修改了同一个引用指向的对象,也就修改了实际参数传来的引用值指向的对象"""

a.append("can change object")

return a

def test():

lst = [1, 2, 3]

print foo(lst) #打印函数返回的列表

print lst

运行test,结果如下:

[1, 2, 3, 'can change object']

[1, 2, 3, 'can change object']

列表lst 在函数foo中被改变了

注意

关于赋值操作

变量赋值

Python中的变量不需要声明,变量的赋值操作既是变量声明和定义的过程。

每个变量在内存中创建,都包括变量的标识,名称和数据这些信息。 每个变量在使用前都必须赋值,变量赋值以后该变量才会被创建。

等号(=)用来给变量赋值。 等号(=)运算符左边是一个变量名,等号(=)运算符右边是存储在变量中的值。

在使用列表的时候,如果直接在foo函数中对a进行赋值,则形参就会指向一个新的内存空间,同时源数据不会改变。

如下:

def foo(a):

"""实际参数传来一个对象[1,2,3]的引用,当时形式参数(局部变量a重新引用到新的对象,也就是说保存了新的对象) 当然不能修改原来的对象了。"""

a = [123, 321] #对 a 直接赋值

return a

def test():

lst = [1, 2, 3]

print foo(lst)

print lst

结果如下:

[123, 321] #函数foo返回的列表

[1, 2, 3]

可以发现,源数据并没有改变。