深入浅析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编程之gui程序实现简单文件浏览器代码
Dec 08 Python
Python实现邮件的批量发送的示例代码
Jan 23 Python
Centos 升级到python3后pip 无法使用的解决方法
Jun 12 Python
Python sklearn KFold 生成交叉验证数据集的方法
Dec 11 Python
flask session组件的使用示例
Dec 25 Python
Python编程深度学习绘图库之matplotlib
Dec 28 Python
Python二叉搜索树与双向链表转换算法示例
Mar 02 Python
Pandas —— resample()重采样和asfreq()频度转换方式
Feb 26 Python
Selenium python时间控件输入问题解决方案
Jul 22 Python
用python 绘制茎叶图和复合饼图
Feb 26 Python
用Python简陋模拟n阶魔方
Apr 17 Python
Python机器学习算法之决策树算法的实现与优缺点
May 13 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
不用iconv库的gb2312与utf-8的互换函数
2006/10/09 PHP
在 PHP 中使用随机数的三个步骤
2006/10/09 PHP
Docker 如何布置PHP开发环境
2016/06/21 PHP
ajax更新数据后,jquery、jq失效问题
2011/03/16 Javascript
推荐11款jQuery开发的复选框和单选框美化插件
2011/08/02 Javascript
jQuery中prepend()方法用法实例
2014/12/25 Javascript
JavaScript中的值类型转换介绍
2014/12/31 Javascript
js变形金刚文字特效代码分享
2015/08/20 Javascript
详解JavaScript函数
2015/12/01 Javascript
js获取本机操作系统类型的两种方法
2015/12/19 Javascript
js实现n秒倒计时后才可以点击的效果
2015/12/20 Javascript
javascript产生随机数方法汇总
2016/01/25 Javascript
解决Mac安装thrift因bison报错的问题
2018/05/17 Javascript
vue实现简单的MVVM框架
2018/08/05 Javascript
每个 JavaScript 工程师都应懂的33个概念
2018/10/22 Javascript
JavaScript使用ul中li标签实现删除效果
2019/04/15 Javascript
JavaScript实现星级评价效果
2019/05/17 Javascript
2020淘宝618理想生活列车自动领喵币js脚本的代码
2020/06/02 Javascript
react-intl实现React国际化多语言的方法
2020/09/27 Javascript
Openlayers绘制地图标注
2020/09/28 Javascript
[47:42]Fnatic vs Liquid 2018国际邀请赛小组赛BO2 第一场 8.16
2018/08/17 DOTA
Python爬虫框架Scrapy安装使用步骤
2014/04/01 Python
Python中不同进制的语法及转换方法分析
2016/07/27 Python
批量将ppt转换为pdf的Python代码 只要27行!
2018/02/26 Python
Pycharm取消py脚本中SQL识别的方法
2018/11/29 Python
Python实现的拉格朗日插值法示例
2019/01/08 Python
python实现五子棋人机对战游戏
2020/03/25 Python
python将dict中的unicode打印成中文实例
2020/05/11 Python
简单了解Django项目应用创建过程
2020/07/06 Python
Python获取excel内容及相关操作代码实例
2020/08/10 Python
Html5适配iphoneX刘海屏的简单实现
2019/04/09 HTML / CSS
巴西独家产品和现场演示购物网站:Shoptime
2019/07/11 全球购物
草莓网官网:StrawberryNET
2019/08/21 全球购物
澳大利亚领先的时尚内衣零售商:Bras N Things
2020/07/28 全球购物
迷你西餐厅创业计划书范文
2013/12/31 职场文书
公司安全管理制度范本
2015/08/05 职场文书