Python参数传递机制传值和传引用原理详解


Posted in Python onMay 22, 2020

首先还是应该科普下函数参数传递机制,传值和传引用是什么意思?

函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题。基本的参数传递机制有两种:值传递和引用传递。

值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本。值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值。

引用传递(pass-by-reference)过程中,被调函数的形式参数虽然也作为局部变量在堆栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过堆栈中存放的地址访问主调函数中的实参变量。正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。

在python中实际又是怎么样的呢?

先看一个简单的例子

from ctypes import *
import os.path 
import sys

def test(c):
  print "test before "
  print id(c)
  c+=2
  print "test after +"
  print id(c)
  return c

def printIt(t):
  for i in range(len(t)):
    print t[i]

if __name__=="__main__":
  a=2
  print "main before invoke test"
  print id(a)
  n=test(a)
  print "main afterf invoke test"
  print a
  print id(a)

运行后结果如下:

>>> 
main before invoke test
test before 
test after +
main afterf invoke test
39601564

id函数可以获得对象的内存地址.很明显从上面例子可以看出,将a变量作为参数传递给了test函数,传递了a的一个引用,把a的地址传递过去了,所以在函数内获取的变量C的地址跟变量a的地址是一样的,但是在函数内,对C进行赋值运算,C的值从2变成了4,实际上2和4所占的内存空间都还是存在的,赋值运算后,C指向4所在的内存。而a仍然指向2所在的内存,所以后面打印a,其值还是2.

如果还不能理解,先看下面例子

>>> a=1
>>> b=1
>>> id(a)
>>> id(b)
>>> a=2
>>> id(a)

a和b都是int类型的值,值都是1,而且内存地址都是一样的,这已经表明了在python中,可以有多个引用指向同一个内存(画了一个很挫的图,见谅),在给a赋值为2后,再次查看a的内存地址,都已经变化了

Python参数传递机制传值和传引用原理详解

而基于最前面的例子,大概可以这样描述:

Python参数传递机制传值和传引用原理详解

那python函数传参就是传引用?然后传参的值在被调函数内被修改也不影响主调函数的实参变量的值?再来看个例子。

from ctypes import *
import os.path 
import sys

def test(list2):
  print "test before "
  print id(list2)
  list2[1]=30
  print "test after +"
  print id(list2)
  return list2

def printIt(t):
  for i in range(len(t)):
    print t[i]

if __name__=="__main__":
  list1=["loleina",25,'female']
  print "main before invoke test"
  print id(list1)
  list3=test(list1)
  print "main afterf invoke test"
  print list1
  print id(list1)

实际值为:

>>> 
main before invoke test
test before 
test after +
main afterf invoke test
['loleina', 30, 'female']

发现一样的传值,而第二个变量居然变化,为啥呢?

实际上是因为python中的序列:列表是一个可变的对象,就基于list1=[1,2] list1[0]=[0]这样前后的查看list1的内存地址,是一样的。

>>> list1=[1,2]
>>> id(list1)
>>> list1[0]=[0]
>>> list1
[[0], 2]
>>> id(list1)

字典也是可变对象:

>>> def fun2(num1,l1,d1):
...   num1=123
...   l1[0]=123
...   d1['a']=123
...   print("inside:","num1=%f,l1=%s,d1=%s"%(num1,l1,d1))
... 
>>> num=111
>>> l=[1,1,1]
>>> d={'a':111,'b':0}
>>> print("before:","num=%f,l=%s,d=%s"%(num,l,d))
before: num=111.000000,l=[1, 1, 1],d={'a': 111, 'b': 0}
>>> fun2(num,l,d)
inside: num1=123.000000,l1=[123, 1, 1],d1={'a': 123, 'b': 0}
>>> print("after:","num=%f,l=%s,d=%s"%(num,l,d))
after: num=111.000000,l=[123, 1, 1],d={'a': 123, 'b': 0}

结论:python不允许程序员选择采用传值还是传引用。Python参数传递采用的肯定是“传对象引用”的方式。这种方式相当于传值和传引用的一种综合。

如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值--相当于通过“传引用”来传递对象。

如果函数收到的是一个不可变对象(比如数字、字符或者元组)的引用,就不能直接修改原始对象--相当于通过“传值'来传递对象。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python代理抓取并验证使用多线程实现
May 03 Python
Python中的下划线详解
Jun 24 Python
python实现网站的模拟登录
Jan 04 Python
selenium+python 对输入框的输入处理方法
Oct 11 Python
对python操作kafka写入json数据的简单demo分享
Dec 27 Python
解决Python中定时任务线程无法自动退出的问题
Feb 18 Python
django 数据库连接模块解析及简单长连接改造方法
Aug 29 Python
简单了解python中的f.b.u.r函数
Nov 02 Python
使用pyplot.matshow()函数添加绘图标题
Jun 16 Python
OpenCV 之按位运算举例解析
Jun 19 Python
python开发入门——set的使用
Sep 03 Python
使用Python爬虫爬取小红书完完整整的全过程
Jan 19 Python
python filecmp.dircmp实现递归比对两个目录的方法
May 22 #Python
关于keras.layers.Conv1D的kernel_size参数使用介绍
May 22 #Python
Python参数传递对象的引用原理解析
May 22 #Python
Python configparser模块常用方法解析
May 22 #Python
keras中的卷积层&池化层的用法
May 22 #Python
Keras Convolution1D与Convolution2D区别说明
May 22 #Python
Python pip安装模块提示错误解决方案
May 22 #Python
You might like
十大“创意”战术!
2020/03/04 星际争霸
php url地址栏传中文乱码解决方法集合
2010/06/25 PHP
smarty模板嵌套之include与fetch性能测试
2010/12/05 PHP
无法在发生错误时创建会话,请检查 PHP 或网站服务器日志,并正确配置 PHP 安装(win+linux)
2012/05/05 PHP
document.onreadystatechange事件的用法分析
2009/10/17 Javascript
几个比较实用的JavaScript 测试及效验工具
2010/04/18 Javascript
jquery操作checked属性以及disabled属性的多种方法
2014/06/20 Javascript
JS中完美兼容各大浏览器的scrolltop方法
2015/04/17 Javascript
javascript实现框架高度随内容改变的方法
2015/07/23 Javascript
jQuery实现列表内容的动态载入特效
2015/08/08 Javascript
JS提交form表单实例分析
2015/12/10 Javascript
jQuery实现点击查看大图并以弹框的形式居中
2016/08/08 Javascript
原生JavaScript实现Tooltip浮动提示框特效
2017/03/07 Javascript
详解探索 vuex 2.0 以及使用 vuejs 2.0 + vuex 2.0 构建记事本应用
2017/06/16 Javascript
详解webpack分离css单独打包
2017/06/21 Javascript
Node.js静态服务器的实现方法
2018/02/28 Javascript
VUE2.0中Jsonp的使用方法
2018/05/22 Javascript
在create-react-app中使用sass的方法示例
2018/10/01 Javascript
vue自定义指令限制输入框输入值的步骤与完整代码
2020/08/30 Javascript
[02:32]【DOTA2亚洲邀请赛】iceice,梦开始的地方
2017/03/13 DOTA
[03:16]DOTA2完美大师赛主赛事首日集锦
2017/11/23 DOTA
[45:17]DOTA2-DPC中国联赛定级赛 Phoenix vs DLG BO3第三场 1月9日
2021/03/11 DOTA
Python中set与frozenset方法和区别详解
2016/05/23 Python
Python数据结构与算法之图的最短路径(Dijkstra算法)完整实例
2017/12/12 Python
Python使用numpy产生正态分布随机数的向量或矩阵操作示例
2018/08/22 Python
python中使用while循环的实例
2019/08/05 Python
Python如何计算语句执行时间
2019/11/22 Python
python使用paramiko实现ssh的功能详解
2020/03/06 Python
python实现横向拼接图片
2020/03/23 Python
英国最大的宠物食品和宠物用品网上零售商: Zooplus
2016/08/01 全球购物
美国男士内衣品牌:Tommy John
2017/12/22 全球购物
宣传标语大全
2014/07/01 职场文书
文化苦旅读书笔记
2015/06/29 职场文书
2016年毕业实习心得体会范文
2015/10/09 职场文书
仅仅使用 HTML/CSS 实现各类进度条的方式汇总
2021/11/11 HTML / CSS
微信小程序APP的生命周期及页面的生命周期
2022/04/19 Javascript