深入理解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 相关文章推荐
Python2.5/2.6实用教程 入门基础篇
Nov 29 Python
python实现分析apache和nginx日志文件并输出访客ip列表的方法
Apr 04 Python
在Python中使用SimpleParse模块进行解析的教程
Apr 11 Python
浅谈Python爬取网页的编码处理
Nov 04 Python
Python3.遍历某文件夹提取特定文件名的实例
Apr 26 Python
Python装饰器的执行过程实例分析
Jun 04 Python
python实现程序重启和系统重启方式
Apr 16 Python
keras 读取多标签图像数据方式
Jun 12 Python
Python脚本破解压缩文件口令实例教程(zipfile)
Jun 14 Python
15个Pythonic的代码示例(值得收藏)
Oct 29 Python
Python APScheduler执行使用方法详解
Dec 10 Python
pycharm 实现复制一行的快捷键
Jan 15 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获取网络上文件
2006/10/09 PHP
PHP网页游戏学习之Xnova(ogame)源码解读(七)
2014/06/23 PHP
基于JQUERY的多级联动代码
2012/01/24 Javascript
JavaScript按位运算符的应用简析
2014/02/04 Javascript
jQuery中parent()方法用法实例
2015/01/07 Javascript
jQuery的position()方法详解
2015/07/19 Javascript
基于jquery实现复选框全选,反选,全不选等功能
2015/10/16 Javascript
ionic实现滑动的三种方式
2016/08/27 Javascript
Bootstrap媒体对象学习使用
2017/03/07 Javascript
bootstrap3-dialog-master模态框使用详解
2017/08/22 Javascript
Node.js中使用mongoose操作mongodb数据库的方法
2017/09/12 Javascript
js实现以最简单的方式将数组元素添加到对象中的方法
2017/12/20 Javascript
jQuery实现判断上传图片类型和大小的方法示例
2018/04/11 jQuery
基于vue-element组件实现音乐播放器功能
2018/05/06 Javascript
JS表格的动态操作完整示例
2020/01/13 Javascript
如何使用Javascript中的this关键字
2020/05/28 Javascript
vue实现两个区域滚动条同步滚动
2020/12/13 Vue.js
[04:38]完美世界携手游戏风云打造 卡尔工作室饰品系统篇
2013/04/25 DOTA
Python实现配置文件备份的方法
2015/07/30 Python
python各种语言间时间的转化实现代码
2016/03/23 Python
Python中getattr函数和hasattr函数作用详解
2016/06/14 Python
python 转换 Javascript %u 字符串为python unicode的代码
2016/09/06 Python
python 列表输出重复值以及对应的角标方法
2019/06/11 Python
使用Python轻松完成垃圾分类(基于图像识别)
2019/07/09 Python
python中count函数简单的实例讲解
2020/02/06 Python
Python获取二维数组的行列数的2种方法
2020/02/11 Python
python可以用哪些数据库
2020/06/22 Python
Python Flask异步发送邮件实现方法解析
2020/08/01 Python
Python高并发和多线程有什么关系
2020/11/14 Python
Python爬虫之Selenium设置元素等待的方法
2020/12/04 Python
社会实践感言
2014/01/25 职场文书
电焊工工作岗位职责
2014/02/06 职场文书
什么是就业协议书
2014/04/17 职场文书
大学生活动总结模板
2014/07/02 职场文书
小学语文教师年度考核个人总结
2015/02/05 职场文书
美国运营商 T-Mobile 以 117.83Mb/s 的速度排第一位
2022/04/21 数码科技