Python基础之值传递和引用传递详解
在python
中,向函数传递参数的类型有两种,一种是值传递,还有一种是引用传递,如果你恰恰好会一点c
基础,你可以理解为前者为传递形参,而后者传递指针。本篇文章将探究python
的值传递和引用传递。
文本所依赖的python
环境为:
什么是值传递和引用传递
值传递,我们可以理解为传递了一个副本过去,即变量的拷贝,修改副本值不会影响原先的值,例如:
def modify_x(x): x = 99 print("函数中修改过后的值: " , x) x=66 modify_x(x) print("执行modify_x函数后的值:" , x)
在上述代码中,我们定义了一个变量x
,并赋值为66
,而后将x
传入其modify_x
函数中,在函数中,我们将x
赋值为99
,打印一下函数中的x
值,函数结果。 在主函数中再打印一下x
的值。
此结果执行后如下:
如上代码,我们传入的是形参,在函数中修改形参是不会改变原先的值的,这是因为函数运行时候会先进行压栈,运行过程中会产线局部信息等,恰恰好,我们传入的形参就是该类型的值,所以运行后会出栈,出栈后函数所在的内存也会被销毁,所以函数内的局部变量随着出栈也被销毁了。所以直接修改形参无效。
以上这个就是值传递。
那什么是引用传递呢?我们还是拿上面这个例子做比方,只不过传递的类型换一下,从数值类型更换为字典类型,如:
def modify_x(dict): dict["x"] = 99 print("数中修改过后的值:" , dict) a={ "x":66 } modify_x(a) print("执行modify_x函数后的值:", a)
如上代码,我们定义了一个字典a
,该字典有一个key
为x
,值为66
。在调用modify_x
函数中,我们将a
传递给了函数,在函数中,我们将该字典key
为x
的赋值为99
,函数结束,在主函数中打印a
的值。
执行后结果如下:
是不是感觉很诧异,同样的代码,为什么传递整形 和 传递 字典 , 所执行的效果不一样呢? 这是因为python
机制就是如此,它在传递该值的时候,使用的是指针传递,所以值没有改变,我们将其称之为引用传递。
可以干预参数传递是值传递还是引用传递么
python
不可以干预参数传递的类型,因为python
不像c
、c++
一样,可以传递形参,也可以传递指针类型。
在python
中,参数传递是由解释器实现的,所以说,普通开发者,没办法直接干预参数传递方式,但是可以曲线救国,善用return
就是其中一条,例如我们将最开始的代码修改一下,不直接修改值,而是返回一个新的值,例如:
def modify_x(x): x = 99 return x x = 66 x = modify_x(x) print("执行modify_x函数后的值:" , x)
我们执行后,结果为:
这并不是修改x
的值,而是接收modify_x
传递回来的新值。
探寻一下值传递底层是如何实现的
我们之前所述的值传递,都是对数据的拷贝,可是现实真的如此么? 我们可以写一个案例来看下:
def modify_x(x): print("函数中:",id(x)) x = 123 print("函数外:",id(x)) modify_x(x)
在上述代码中,有一个新的知识点是方法id
,它可以查看变量的内存地址。在上述例子中,在主函数中定义一个整形x
,值为123
,在传递给函数前,使用id
方法查看一下变量的内存地址。而后传递给函数modify_x
,在该函数中,也使用id
方法来查看一下形参x
的地址。
若真如我们所猜想,那么2个内存地址应该不一致才对,我们运行下程序:
发现函数内,和函数外的地址都是一样的? 哎,这是怎么回事呢?
这是因为在python
中,解释器为了优化性能,避免大量无用数据拷贝,所以在传递的时候,一开始全是传递的实参,只有当函数内修改了值后,才会新申请一个内存来存该值。细节可以查看这个例子:
def modify_x(x) : print("函数中1:",id(x)) x = 456 print("函数中2:",id(x)) x=123 print("函数外0:",id(x)) modify_x(x)
上述代码,我们在modify_x
函数中,修改变量x
前后都打印其内存地址,结果如下:
我们发现,在未修改之前,地址内存都是指向同一个地址,修改之后,内存地址也变了。
如果我们将x
更换为引用传递的数据的话,就不会出现以下这种情况,可以看下面这个例子:
def modify x(x) : print("函数中1:",id(x)) x[0] = 456 print("函数中2:",id(x)) x = [123] print("函数外0:",id(x)) modify_x(x) print("最后的值:" , x)
上述代码,我们做了一个小小的改动,我们将整形数据x
,更改为了列表类型,最后再打印一下x
的值,查看变了没有,代码运行结果如下:
发现内存地址的值并没有改变,且x
的值在函数中真的被修改了。
所以通过上述例子,可以说明,值传递的时候,再没有修改的时候,该变量地址还是指向原来的地址,当值被修改后,就会开辟一个新的内存地址用于存储该值。这样的话可以避免拷贝大量数据。
最后再总结一下,哪些类型是引用传递,哪些类型是值传递:
引用传递分别有 列表、字典、集合、自定义类实例等。
值传递分别有 字符串类型、元组、布尔类型、数值类型等。
总结
本篇文章简单介绍了值传递和引用传递,值传递,修改函数内值后,不会影响原始值,而引用传递,修改函数内值后,会印象到原始数据。不过有一个小细节,就是值传递,若不进行修改值的时候,其实内存地址是指向的原始值的地址,当修改值的时候,才会真正申请内存来存储修改的值,但是随着函数出栈,该函数内的数据局部变量,也会被销毁。
到此这篇关于Python基础之值传递和引用传递详解的文章就介绍到这了,更多相关Python值传递 引用传递内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
关于torch.optim的灵活使用详解(包括重写SGD,加上L1正则)
今天小编就为大家分享一篇关于torch.optim的灵活使用详解(包括重写SGD,加上L1正则),具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧2020-02-02Pytorch如何加载自己的数据集(使用DataLoader读取Dataset)
这篇文章主要介绍了Pytorch如何加载自己的数据集(使用DataLoader读取Dataset)问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教2022-12-12
最新评论