深入浅析Python传值与传址


Posted in Python onJuly 10, 2018

1. 传值与传址的区别

传值就是传入一个参数的值,传址就是传入一个参数的地址,也就是内存的地址(相当于指针)。他们的区别是如果函数里面对传入的参数重新赋值,函数外的全局变量是否相应改变:用传值传入的参数是不会改变的,用传址传入就会。

def a(n):
  n[2] = 100
  print(n)
  return None
def b(n):
  n += 100
  print(n)
  return None
an = [1,2,3,4,5]
bn = 10
print(an)
a(an)
print(an)
print(bn)
b(bn)
print(bn)
[1, 2, 3, 4, 5]
[1, 2, 100, 4, 5]
[1, 2, 100, 4, 5]
10
110
10

在上面的例子中,an是一个list,将其作为实参传入函数a中,a对其第三个元素进行修改。a执行结束后再次打印an,发现里面的元素的确发生变化,这就是传址操作。bn代表一个数字,将其传入函数b,并做修改,b执行结束后再次打印bn,没有变化,这是传值操作。

2. Python中传值与传址的规律

Python是不允许程序员选择采用传值还是传址的。Python参数传递采用的是“传对象引用”的方式,实际上,这种方式相当于传值和传址的一种综合。

如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值——相当于传址。如果函数收到的是一个不可变对象(比如数字、字符或者元组)的引用,就不能直接修改原始对象——相当于传值。所以python的传值和传址是根据传入参数的类型来选择的。

传值的参数类型:数字,字符串,元组

传址的参数类型:列表,字典

3. 内置函数id

内置函数id,负责显示一个变量或者数据在内存中的地址,有时可以用来检测所使用的对象是否为同一个,帮助区别传值与传址操作。

但是id在有些情况下比较特殊,注意下面的例子。

a = 100
b = 200 
print(id(a))
print(id(b))
c = a
print(id(c))
print(a is c)
a += 300
print(a)
print(c)
print(a is c)
print(id(a))
print(id(c))
1549495552
1549498752
1549495552
True
400
100
False
93638128
1549495552

为了提高内存利用效率,对于一些简单的对象,如一些数值较小的int对象,python采取重用对象内存的办法。如指向a=100,c=100时,由于100作为简单的int类型且数值小,python不会两次为其分配内存,而是只分配一次,然后将a与c同时指向已分配的对象。但是当a的值发生变化时,会单独为a重新分配一个新的内存。

4. list传值与传址

list类型使用简单的赋值操作,是传址。

a = [1,2,3,4,5]
b = a
print(a)
b[2] = 333
print(a)
print(b)
print(id(a))
print(id(b))
[1, 2, 3, 4, 5]
[1, 2, 333, 4, 5]
[1, 2, 333, 4, 5]
96142472
96142472

copy函数是浅拷贝,是传值。python2中,需要import copy模块,python3可直接使用。

a = [1,2,3,4,5]
b = a.copy()
print(a)
b[2] = 333
print(a)
print(b)
print(id(a))
print(id(b))
[1, 2, 3, 4, 5]
[1, 2, 3, 4, 5]
[1, 2, 333, 4, 5]
92990536
96202632

由于copy是浅拷贝,只拷贝一层的内容,当遇到下列情况时,copy不能实现完全的传值操作。

a = [1,2,3,[10,20,30]]
b = a.copy()
print(id(a))
print(id(b))
print(id(a[3]))
print(id(b[3]))
a[3][2] = 666
print(a)
print(b)
96141704
93355400
96141768
96141768
[1, 2, 3, [10, 20, 666]]
[1, 2, 3, [10, 20, 666]]

要解决这个问题,需要使用deepcopy。python3中,直接可以使用copy()方法,但deepcopy()还是需要导入copy模块。

import copy
a = [1,2,3,[10,20,30]]
b = copy.deepcopy(a)
print(id(a))
print(id(b))
print(id(a[3]))
print(id(b[3]))
a[3][2] = 666
print(a)
print(b)
96503944
93002376
96886024
93352712
[1, 2, 3, [10, 20, 666]]
[1, 2, 3, [10, 20, 30]]

5. tuple操作

tuple元组是不可修改的,指的是其元组内容不可改。

t1 = (1,2,3)
t1[1] = 100
---------------------------------------------------------------------------
TypeError                 Traceback (most recent call last)
<ipython-input-19-9caf76a526a9> in <module>()
   1 t1 = (1,2,3)
----> 2 t1[1] = 100
TypeError: 'tuple' object does not support item assignment

但是其所指向的内存地址是可变的。

t1 = (1,2,3)
t2 = (5,6,7)
print(id(t1))
t1 += t2
print(t1)
print(id(t1))
print(id(t2))
t2 *= 3
print(t2)
print(id(t2))
96151520
(1, 2, 3, 5, 6, 7)
93048552
94080672
(5, 6, 7, 5, 6, 7, 5, 6, 7)
93656912

并不是起初的t1和t2所指向的元组内容发生了变化,而是新分配了两个元组内存,t1和t2所指向的内存发生改变。

总结

Python 相关文章推荐
python实现simhash算法实例
Apr 25 Python
详解Python中的type()方法的使用
May 21 Python
Python中线程编程之threading模块的使用详解
Jun 23 Python
python subprocess 杀掉全部派生的子进程方法
Jan 16 Python
Django实现分页功能
Jul 02 Python
替换python字典中的key值方法
Jul 06 Python
python安装pywin32clipboard的操作方法
Jan 24 Python
pytorch使用Variable实现线性回归
May 21 Python
python英语单词测试小程序代码实例
Sep 09 Python
pygame实现烟雨蒙蒙下彩虹雨
Nov 11 Python
python 实现简单的FTP程序
Dec 27 Python
windows支持哪个版本的python
Jul 03 Python
Python+OpenCV目标跟踪实现基本的运动检测
Jul 10 #Python
python3读取excel文件只提取某些行某些列的值方法
Jul 10 #Python
python读取excel指定列数据并写入到新的excel方法
Jul 10 #Python
python 常用的基础函数
Jul 10 #Python
使用pandas批量处理矢量化字符串的实例讲解
Jul 10 #Python
python opencv实现运动检测
Jul 10 #Python
python中单下划线_的常见用法总结
Jul 10 #Python
You might like
用Flash图形化数据(一)
2006/10/09 PHP
php内核解析:PHP中的哈希表
2014/01/30 PHP
PHPExcel读取EXCEL中的图片并保存到本地的方法
2015/02/14 PHP
thinkPHP学习笔记之安装配置篇
2015/03/05 PHP
php中smarty变量修饰用法实例分析
2015/06/11 PHP
最新最全PHP生成制作验证码代码详解(推荐)
2016/06/12 PHP
php使用curl实现简单模拟提交表单功能
2017/05/15 PHP
PHP有序表查找之二分查找(折半查找)算法示例
2018/02/09 PHP
硬盘浏览程序,保存成网页格式便可使用
2006/12/03 Javascript
基于jQuery的左右滚动实现代码
2010/12/03 Javascript
js 中{},[]中括号,大括号使用详解
2011/05/12 Javascript
jQuery中实现动画效果的基本操作介绍
2013/04/16 Javascript
node.js中的fs.chownSync方法使用说明
2014/12/16 Javascript
修改或扩展jQuery原生方法的代码实例
2015/01/13 Javascript
jquery点击展示与隐藏更多内容
2016/12/03 Javascript
JavaScript事件委托原理与用法实例分析
2018/06/07 Javascript
分享vue里swiper的一些坑
2018/08/30 Javascript
微信小程序实现选项卡效果
2018/11/06 Javascript
微信小程序如何自定义table组件
2019/06/29 Javascript
[06:37]2014DOTA2国际邀请赛 昔日王者渴望重回巅峰
2014/07/12 DOTA
使用python遍历指定城市的一周气温
2017/03/31 Python
Pycharm编辑器技巧之自动导入模块详解
2017/07/18 Python
利用python实现逐步回归
2020/02/24 Python
pyMySQL SQL语句传参问题,单个参数或多个参数说明
2020/06/06 Python
uniapp+Html5端实现PC端适配
2020/07/15 HTML / CSS
新东网科技Java笔试题
2012/07/13 面试题
优秀学生获奖感言
2014/02/15 职场文书
2014年秋季开学典礼主持词
2014/08/02 职场文书
学生会竞选演讲稿纪检部
2014/08/25 职场文书
纪念九一八事变演讲稿:勿忘国耻
2014/09/14 职场文书
行政前台岗位职责
2015/04/16 职场文书
2016年社区“6.26”禁毒日宣传活动总结
2016/04/05 职场文书
导游词之上海豫园
2019/10/24 职场文书
Golang之sync.Pool使用详解
2021/05/06 Golang
剑指Offer之Java算法习题精讲二叉树的构造和遍历
2022/03/21 Java/Android
分享五个Node.js开发的优秀实践 
2022/04/07 NodeJs