十年网站开发经验 + 多家企业客户 + 靠谱的建站团队
量身定制 + 运营维护+专业推广+无忧售后,网站问题一站解决
首先还是应该科普下函数参数传递机制,传值和传引用是什么意思?
为佛山等地区用户提供了全套网页设计制作服务,及佛山网站建设行业解决方案。主营业务为成都做网站、成都网站设计、成都外贸网站建设、佛山网站设计,以传统方式定制建设网站,并提供域名空间备案等一条龙服务,秉承以专业、用心的态度为用户提供真诚的服务。我们深信只要达到每一位用户的要求,就会得到认可,从而选择与我们长期合作。这样,我们也可以走得更远!
函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题。基本的参数传递机制有两种:值传递和引用传递。
值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本。值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值。
引用传递(pass-by-reference)过程中,被调函数的形式参数虽然也作为局部变量在堆栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过堆栈中存放的地址访问主调函数中的实参变量。正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。
在python中实际又是怎么样的呢?
先看一个简单的例子:
from ctypes import *import os.path
import sysdef test(c): print "test before "
print id(c)
c+=2 print "test after +"
print id(c) return cdef printIt(t): for i in range(len(t)): print t[i]if __name__=="__main__":
a=2 print "main before invoke test"
print id(a)
n=test(a) print "main afterf invoke test"
print a print id(a)
运行后结果如下:
main before invoke test39601564test before
39601564test after +
39601540main afterf invoke test2
39601564
id函数可以获得对象的内存地址.很明显从上面例子可以看出,将a变量作为参数传递给了test函数,传递了a的一个引用,把a的地址传递过去了,所以在函数内获取的变量C的地址跟变量a的地址是一样的,但是在函数内,对C进行赋值运算,C的值从2变成了4,实际上2和4所占的内存空间都还是存在的,赋值运算后,C指向4所在的内存。而a仍然指向2所在的内存,所以后面打印a,其值还是2.
如果还不能理解,先看下面例子
a=1
b=1
id(a)
40650152
id(b)
40650152
a=2
id(a)
40650140
a和b都是int类型的值,值都是1,而且内存地址都是一样的,这已经表明了在python中,可以有多个引用指向同一个内存(画了一个很挫的图,见谅),在给a赋值为2后,再次查看a的内存地址,都已经变化了
而基于最前面的例子,大概可以这样描述:
那python函数传参就是传引用?然后传参的值在被调函数内被修改也不影响主调函数的实参变量的值?再来看个例子。
from ctypes import *import os.path
import sysdef test(list2): print "test before "
print id(list2)
list2[1]=30 print "test after +"
print id(list2) return list2def printIt(t): for i in range(len(t)): print t[i]if __name__=="__main__":
list1=["loleina",25,'female'] print "main before invoke test"
print id(list1)
list3=test(list1) print "main afterf invoke test"
print list1 print id(list1)
实际值为:
main before invoke test64129944test before
64129944test after +
64129944main afterf invoke test
['loleina', 30, 'female']64129944
发现一样的传值,而第二个变量居然变化,为啥呢?
实际上是因为python中的序列:列表是一个可变的对象,就基于list1=[1,2] list1[0]=[0]这样前后的查看list1的内存地址,是一样的。
list1=[1,2] id(list1)64185208
list1[0]=[0] list1
[[0], 2] id(list1)64185208
结论:python不允许程序员选择采用传值还是传引用。Python参数传递采用的肯定是“传对象引用”的方式。这种方式相当于传值和传引用的一种综合。如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值--相当于通过“传引用”来传递对象。如果函数收到的是一个不可变对象(比如数字、字符或者元组)的引用,就不能直接修改原始对象--相当于通过“传值'来传递对象。
Python变量没有赋值,都是引用。
大多语言,是声明一个变量,给它分配一个空间保存一个值,也就是赋值。
Python则是给一个值分配一个空间,变量=这个值,只是这个变量引用了这个值的地址,也就是说,a=1,b=1,c=1,Python只分配了一个空间,保存这个值1。 a,b,c都引用了这个地址。
至于为什么列表的值更改了,因为列表是可变变量
对于可变参数默认是引用传值, 但是不能去修改它的指向, 一旦修改就是按值传递.
# coding=utf-8
def f(a):
a = [0]
print(a)
if __name__ == '__main__':
a = [1, 2, 3]
f(a)
print(a)
上面的代码对a重新赋值, 试图改变a的指向, 那么这时的a就是一个新的局部变量, 而非全局变量a
像a[0] = 100, a.append(0)的操作不会触发上述规则, 和你的输出一样
Python函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题。基本的参数传递
机制有两种:值传递和引用传递。值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开
辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本。值传递的特点是被调函数对形式参数的任何操作都是作
为局部变量进行,不会影响主调函数的实参变量的值。(推荐学习:Python视频教程)
引用传递(pass-by-reference)过程中,被调函数的形式参数虽然也作为局部变量在堆栈中开辟了内存空间,但是这时存放的是由主调函
数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过堆栈中存放的地址访问主调函数中的实参变量。正
因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。
看datetime()函数的原型:
其中前三个参数year, month, day是位置参数,因此传参时可以写参数名,也可以不写参数名。
datetime(2020, 1, 1)
datetime(2020, month=1, day=1)
datetime(year=2020, month=1, day=1)
以上这些传参方式均合法。