深入理解python中函数传递参数是值传递还是引用传递


Posted in Python onNovember 07, 2017

目前网络上大部分博客的结论都是这样的:

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

你可以在很多讨论该问题的博客里找到以上这一段话。

但是在实际操作中我却发现一个问题:

l=[1,2,3]
def a(x):
  x=x+[4]
a(l)
print(l)

这段代码的输出为:

[1,2,3]

为什么是这样呢,list是可变对象,按照上面的结论来说传递方式是引用传递,我应该在函数里能对它进行修改呀?难道不应该输出[1,2,3,4]吗?

我觉得我上面引用的那段大多数博主的结论,其实非常不好理解,而且没有讲到本质,看的云里雾里的。

经过我后面的多次试验,得到以下结论:

其实在python中讨论值传递还是引用传递是没有意义的,要真正对这些情况作出解释,其实是应该搞明白python(对可变对象和不可变对象的)赋值过程中是如何分配内存地址的。

接下来,我们不讨论值传递和引用传递的问题。

让我们做一个非常简单的小实验,其中,id()可以查看变量在内存中的地址:

l1=[1,2,3]
l2=[1,2,3]
a=1
b=1
print(id(l1))
print(id(l2))
print(id(a))
print(id(b))

在我的电脑中的运行结果:

12856594504
12856915080
1643643344
1643643344

可以发现,对于可变对象list来说,即便列表内容一模一样,python也会给它们分配新的不同的地址。

然而,对于不可变对象int来说,内存里只有一个1。即便再定义一个变量c=1,也是指向内存中同一个1。换句话说,不可变对象1的地址是共享的。

接下来让我们看看在函数中调用可变对象和不可变对象,并修改他们的值,会是一个什么情况。

对于不可变对象int,我们来看看最简单的情况:

a=1
print(id(a))
def x(a):
  print(id(a))
  b=a
  print(id(b))
x(a)

运行得到:

1643643344
1643643344
1643643344

这看起来就是一个引用传递,函数外的a、函数里的a和b都指向了同一个地址。

但我们再来看一个极端情况:

a=1
print(id(a))
def x():
  b=1
  print(id(b))
x()

运行得到:

1643643344
1643643344

很神奇不是吗?函数外定义的a和函数内定义的b没有任何关系,但它们指向同一个地址!

所以你说如何判断它是值传递还是引用传递?讨论这个问题根本没有意义,因为内存里只有一个1。当我把值1传递给函数里的某一个变量的时候,我实际上也传递了地址,因为内存里只有一个1。

甚至于说我直接给函数里的b赋值1都可以让函数外的a和函数内的b指向同一个地址。

下面来看看传递可变对象list的情况:

l=[1,2,3]
print(id(l))
def a(x):
  print(id(x))
  x.pop()
  print(x)
  print(id(x))
  x=x+[3]
  print(x)
  print(id(x))
a(l)

运行得到

883142451528
[1, 2]
[1, 2, 3]

可以看到,当我们把函数外的列表L传递给函数后,x的地址和L是一样的,这看起来就是一个引用传递,没问题。

继续往下,我们调用x本身的方法pop后,x变成[1,2],并且x的地址没变,这也没什么问题。

但是当我们给x赋值以后,x的地址就变了。

也就是说,只要创建一个新的可变对象,python就会分配一个新的地址。就算我们创建的新可变对象和已存在的旧可变对象完全一样,python依旧会分配一个新的地址(见本文上半部分那个‘非常简单的小实验')

而pop并不是创建新的可变对象,pop是对已有的可变对象进行修改。

所以可以总结为:

在python中,不可变对象是共享的,创建可变对象永远是分配新地址

这个时候我们再回过头来思考值传递和引用传递的问题,就会发现在python里讨论这个确实是没有意义。

我们可以说:python有着自己的一套特殊的传参方式,这是由python动态语言的性质所决定的

总结

以上就是本文关于深入理解python中函数传递参数是值传递还是引用传递的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站:Python实现一个简单的验证码程序、Python编程django实现同一个ip十分钟内只能注册一次、简单了解Python中的几种函数等,有什么问题可以随时留言,小编会及时回复大家的。感谢朋友们对本站的支持!

Python 相关文章推荐
详解使用Python处理文件目录的相关方法
Oct 16 Python
[原创]windows下Anaconda的安装与配置正解(Anaconda入门教程)
Apr 05 Python
Python subprocess库的使用详解
Oct 26 Python
python 利用for循环 保存多个图像或者文件的实例
Nov 09 Python
对Python3 解析html的几种操作方式小结
Feb 16 Python
python把1变成01的步骤总结
Feb 27 Python
详解Python3 pandas.merge用法
Sep 05 Python
Python使用matplotlib 模块scatter方法画散点图示例
Sep 27 Python
OpenCV哈里斯(Harris)角点检测的实现
Jan 15 Python
简单了解如何封装自己的Python包
Jul 08 Python
JupyterNotebook 输出窗口的显示效果调整实现
Sep 22 Python
Python解析m3u8拼接下载mp4视频文件的示例代码
Mar 03 Python
python中numpy.zeros(np.zeros)的使用方法
Nov 07 #Python
django项目运行因中文而乱码报错的几种情况解决
Nov 07 #Python
Python创建二维数组实例(关于list的一个小坑)
Nov 07 #Python
python 简单备份文件脚本v1.0的实例
Nov 06 #Python
Python如何实现MySQL实例初始化详解
Nov 06 #Python
django rest framework之请求与响应(详解)
Nov 06 #Python
基于python中的TCP及UDP(详解)
Nov 06 #Python
You might like
php二分法在IP地址查询中的应用
2008/08/12 PHP
php继承的一个应用
2011/09/06 PHP
php图片缩放实现方法
2014/02/20 PHP
php获取微信基础接口凭证Access_token
2018/08/23 PHP
工作需要写的一个js拖拽组件
2011/07/28 Javascript
javascript自动恢复文本框点击清除后的默认文本
2016/01/12 Javascript
多个上传文件用js验证文件的格式和大小的方法(推荐)
2017/03/09 Javascript
JavaScript实现的冒泡排序法及统计相邻数交换次数示例
2017/04/26 Javascript
bootstrap paginator分页前后台用法示例
2017/06/17 Javascript
详解webpack2异步加载套路
2018/09/14 Javascript
nodejs异步编程基础之回调函数用法分析
2018/12/26 NodeJs
微信小程序错误this.setData报错及解决过程
2019/09/18 Javascript
利用layer实现表单完美验证的方法
2019/09/26 Javascript
JS立即执行的匿名函数用法分析
2019/11/04 Javascript
基于vue的tab-list类目切换商品列表组件的示例代码
2020/02/14 Javascript
JS addEventListener()和attachEvent()方法实现注册事件
2021/01/11 Javascript
使用JS实现鼠标放上图片进行放大离开实现缩小功能
2021/01/27 Javascript
Python中的time模块与datetime模块用法总结
2016/06/30 Python
Python调用adb命令实现对多台设备同时进行reboot的方法
2018/10/15 Python
Python中logging实例讲解
2019/01/17 Python
python 缺失值处理的方法(Imputation)
2019/07/02 Python
Flask框架钩子函数功能与用法分析
2019/08/02 Python
python实现文件的分割与合并
2019/08/29 Python
python调用Matplotlib绘制分布点图
2019/10/18 Python
python3将变量写入SQL语句的实现方式
2020/03/02 Python
俄罗斯在线手表和珠宝商店:AllTime
2019/09/28 全球购物
介绍一下JNDI的基本概念
2013/07/26 面试题
描述JSP和Servlet的区别、共同点、各自应用的范围
2012/10/02 面试题
三好学生自我鉴定
2013/12/17 职场文书
日语求职信范文
2013/12/17 职场文书
历史专业学生的自我评价
2014/02/28 职场文书
小学秋季运动会报道稿
2014/09/30 职场文书
公路施工安全责任书
2015/05/08 职场文书
先进个人事迹材料(2016推荐版)
2016/03/01 职场文书
为什么node.js不适合大型项目
2021/04/28 Javascript
Java数组与堆栈相关知识总结
2021/06/29 Java/Android