深入浅析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实现保存网页到本地示例
Mar 16 Python
Python实现去除代码前行号的方法
Mar 10 Python
django框架如何集成celery进行开发
May 24 Python
flask框架使用orm连接数据库的方法示例
Jul 16 Python
Django添加feeds功能的示例
Aug 07 Python
Python面向对象程序设计构造函数和析构函数用法分析
Apr 12 Python
Python3 实现串口两进程同时读写
Jun 12 Python
python图片剪裁代码(图片按四个点坐标剪裁)
Mar 10 Python
解决jupyter notebook显示不全出现框框或者乱码问题
Apr 09 Python
python字符串拼接+和join的区别详解
Dec 03 Python
python使用XPath解析数据爬取起点小说网数据
Apr 22 Python
解决Python字典查找报Keyerror的问题
May 26 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
PHP新手上路(五)
2006/10/09 PHP
php准确获取文件MIME类型的方法
2015/06/17 PHP
PHP YII框架开发小技巧之模型(models)中rules自定义验证规则
2015/11/16 PHP
深入理解PHP JSON数组与对象
2016/07/19 PHP
基于ThinkPHP5.0实现图片上传插件
2017/09/25 PHP
Laravel中unique和exists验证规则的优化详解
2018/01/28 PHP
零基础php编程好学吗
2019/10/11 PHP
DWZ table的原生分页浅谈
2013/03/01 Javascript
Javascript 读取操作Sql中的Xml字段
2014/10/09 Javascript
Javascript冒泡排序算法详解
2014/12/03 Javascript
浅谈 javascript 事件处理
2015/01/04 Javascript
js实现简单排列组合的方法
2016/01/27 Javascript
JavaScript学习笔记之数组随机排序
2016/03/23 Javascript
javascript操作cookie
2017/01/17 Javascript
Vue+webpack+Element 兼容问题总结(小结)
2018/08/16 Javascript
4 种滚动吸顶实现方式的比较
2019/04/09 Javascript
Node.js如何优雅的封装一个实用函数的npm包的方法
2019/04/29 Javascript
jquery制作的移动端购物车效果完整示例
2020/02/24 jQuery
微信小程序12行js代码自己写个滑块功能(推荐)
2020/07/15 Javascript
vue实现简单加法计算器
2020/10/22 Javascript
python 中if else 语句的作用及示例代码
2018/03/05 Python
java判断三位数的实例讲解
2019/06/10 Python
python GUI库图形界面开发之PyQt5开发环境配置与基础使用
2020/02/25 Python
Node.js 和 Python之间该选择哪个?
2020/08/05 Python
通过Python pyecharts输出保存图片代码实例
2020/11/25 Python
python requests库的使用
2021/01/06 Python
日本最大的眼镜购物网站:Oh My Glasses
2016/11/13 全球购物
Andrew Marc官网:设计师外套的领先制造商
2019/10/30 全球购物
投资合作协议书
2014/04/17 职场文书
岗位说明书范文
2014/05/07 职场文书
安全生产宣传标语
2014/06/06 职场文书
神秘岛读书笔记
2015/07/01 职场文书
七年级作文之我的梦想
2019/10/16 职场文书
有趣的二维码:使用MyQR和qrcode来制作二维码
2021/05/10 Python
MySQL COUNT函数的使用与优化
2021/05/10 MySQL
十大动画制作软件,Adobe产品上榜两款,第一是行业标准软件
2022/03/18 杂记