python函数中参数的传递

 Python唯一支持的参数传递方式是『共享传参』(call by sharing)多数面向对象语言都采用这一模式,包括Ruby、Smalltalk和Java(Java的引用类型是这样,基本类型按值传递)共享传参是指函数的各个形式参数获得实参中各个引用的副本;也就是说,函数内部的形参是实参的别名(alias)这种方案的结果是,函数可能会修改作为参数传入的可变对象,但是无法修改那些对象的标识(即不能把一个对象替换为另一个对象.

 1  >>> def fun(num,l,d):
 2 ...     num=123;
 3 ...     l=[1,2,3]
 4 ...     d={'a':123}
 5 ...     print("inside:","num=%f,l=%s,d=%s"%(num,l,d))
 6 ... 
 7 >>> num=1
 8 >>> l=[1,1,1]
 9 >>> d={'nice':111}
10 >>> print("before:","num=%f,l=%s,d=%s"%(num,l,d))
11 before: num=1.000000,l=[1, 1, 1],d={'nice': 111}
12 >>> fun(num,l,d)
13 inside: num=123.000000,l=[1, 2, 3],d={'a': 123}
14 >>> print("after:","num=%f,l=%s,d=%s"%(num,l,d))
15 after: num=1.000000,l=[1, 1, 1],d={'nice': 111
需要说明的是:函数内部的num,l,d和命令行里的num,l,d是不同的变量,只是名字相同。函数内部的是形参,命令行里的是实参。
修改传递进来对象的内容,即不是让形参指向不同对象,而是通过引用修改对象内容。当然这个对象必须是可变的。
 1 >>> def fun2(num1,l1,d1):
 2 ...     num1=123
 3 ...     l1[0]=123
 4 ...     d1['a']=123
 5 ...     print("inside:","num1=%f,l1=%s,d1=%s"%(num1,l1,d1))
 6 ... 
 7 >>> num=111c
 8 >>> l=[1,1,1]
 9 >>> d={'a':111,'b':0}
10 >>> print("before:","num=%f,l=%s,d=%s"%(num,l,d))
11 before: num=111.000000,l=[1, 1, 1],d={'a': 111, 'b': 0}
12 >>> fun2(num,l,d)
13 inside: num1=123.000000,l1=[123, 1, 1],d1={'a': 123, 'b': 0}
14 >>> print("after:","num=%f,l=%s,d=%s"%(num,l,d))
15 after: num=111.000000,l=[123, 1, 1],d={'a': 123, 'b': 0}

小结:python中一切皆对象,函数中传递的是对象的引用,当形参指向了不同对象,实参不会改变;当形参通过传递来的引用修改了对象的内容,实参会跟着改变,因为形参和实参指向的是同一个对象。其中不仅是函数参数的传递是这样,在函数内部也是这样。当一个变量向另一个变量赋值时,给的是对象的引用,这个对象的引用,感觉相当于c语言中变量的地址,只是在python中,这个地址变量是通用的,可以指向任意类型,当你改变了这个地址,相当与python中指向了不同对象,原来变量的地址没有改变,所以原来变量指向的地址的内容没有改变。当你通过这个地址改变了地址里面的内容时,原来变量指向的地址的内容也会跟着改变。



猜你喜欢

转载自www.cnblogs.com/qq991025/p/11691370.html