最近学习用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]
可以发现,源数据并没有改变。