python 函数传参之传值还是传引用的分析

作者:loleina 时间:2021-08-20 13:06:34 

首先还是应该科普下函数参数传递机制,传值和传引用是什么意思?

函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题。基本的参数传递机制有两种:值传递和引用传递。

值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本。值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值。

引用传递(pass-by-reference)过程中,被调函数的形式参数虽然也作为局部变量在堆栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过堆栈中存放的地址访问主调函数中的实参变量。正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。

在python中实际又是怎么样的呢?

先看一个简单的例子:


from ctypes import *
import os.path
import sys

def test(c):
 print "test before "
 print id(c)
 c+=2
 print "test after +"
 print id(c)
 return c

def 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 test
test before
test after +
main afterf invoke test
39601564

d函数可以获得对象的内存地址.很明显从上面例子可以看出,将a变量作为参数传递给了test函数,传递了a的一个引用,把a的地址传递过去了,所以在函数内获取的变量C的地址跟变量a的地址是一样的,但是在函数内,对C进行赋值运算,C的值从2变成了4,实际上2和4所占的内存空间都还是存在的,赋值运算后,C指向4所在的内存。而a仍然指向2所在的内存,所以后面打印a,其值还是2.

如果还不能理解,先看下面例子


>>> a=1
>>> b=1
>>> id(a)
>>> id(b)
>>> a=2
>>> id(a)

python 函数传参之传值还是传引用的分析

而基于最前面的例子,大概可以这样描述:

python 函数传参之传值还是传引用的分析

那python函数传参就是传引用?然后传参的值在被调函数内被修改也不影响主调函数的实参变量的值?再来看个例子。


from ctypes import *
import os.path
import sys

def test(list2):
 print "test before "
 print id(list2)
 list2[1]=30
 print "test after +"
 print id(list2)
 return list2

def 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 test
test before
test after +
main afterf invoke test
['loleina', 30, 'female']

发现一样的传值,而第二个变量居然变化,为啥呢?

实际上是因为python中的序列:列表是一个可变的对象,就基于list1=[1,2] list1[0]=[0]这样前后的查看list1的内存地址,是一样的。


>>> list1=[1,2]
>>> id(list1)
>>> list1[0]=[0]
>>> list1
[[0], 2]
>>> id(list1)

结论:python不允许程序员选择采用传值还是传引用。Python参数传递采用的肯定是“传对象引用”的方式。这种方式相当于传值和传引用的一种综合。如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值--相当于通过“传引用”来传递对象。如果函数收到的是一个不可变对象(比如数字、字符或者元组)的引用,就不能直接修改原始对象--相当于通过“传值'来传递对象。

来源:http://www.cnblogs.com/loleina/p/5276918.html

标签:python,函数传参
0
投稿

猜你喜欢

  • 解析python高级异常和运算符重载

    2021-06-17 00:56:41
  • 《悟透JavaScript》感谢语

    2008-11-12 12:59:00
  • Pyqt5 实现窗口缩放,控件在窗口内自动伸缩的操作

    2022-10-16 06:32:09
  • jupyter notebook 调用环境中的Keras或者pytorch教程

    2022-10-07 23:23:09
  • Python面向对象编程中的类和对象学习教程

    2023-08-14 06:47:30
  • golang中的空slice案例

    2023-09-02 12:26:36
  • python中yield的用法详解——最简单,最清晰的解释

    2021-10-22 21:27:56
  • MySQL如何利用DCL管理用户和控制权限

    2024-01-14 13:33:21
  • Python元组常见操作示例

    2023-06-28 20:48:26
  • 在python中logger setlevel没有生效的解决

    2021-12-13 16:03:01
  • JS图片根据鼠标滚动延时加载的实例代码

    2024-04-22 13:22:33
  • vue解决跨域问题的几种常用方法(CORS)

    2024-04-28 09:32:42
  • asp利用xmlhttp抓取特定网页内容例子

    2008-10-10 12:58:00
  • python列表逆序排列的4种方法

    2022-09-19 18:26:21
  • Vue ElementUI在el-table中使用el-popover问题

    2024-05-09 15:22:15
  • python变量命名的7条建议

    2021-05-15 20:16:07
  • 用Dreamweaver MX制作导航下拉菜单

    2009-05-29 18:37:00
  • 亚马逊购物用户体验分析 (一)

    2009-10-25 12:40:00
  • 解决pycharm的Python console不能调试当前程序的问题

    2021-09-08 12:02:42
  • MySQL数据库改名的详细方法教程

    2024-01-18 11:46:27
  • asp之家 网络编程 m.aspxhome.com