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分割和拼接字符串
Nov 01 Python
解决python升级引起的pip执行错误的问题
Jun 12 Python
python 读取目录下csv文件并绘制曲线v111的方法
Jul 06 Python
解决在pycharm中显示额外的 figure 窗口问题
Jan 15 Python
GitHub 热门:Python 算法大全,Star 超过 2 万
Apr 29 Python
Python-接口开发入门解析
Aug 01 Python
python语言线程标准库threading.local解读总结
Nov 10 Python
python给图像加上mask,并提取mask区域实例
Jan 19 Python
浅谈keras2 predict和fit_generator的坑
Jun 17 Python
使用Keras实现Tensor的相乘和相加代码
Jun 18 Python
python中get和post有什么区别
Jun 19 Python
python和Appium的移动端多设备自动化测试框架
Apr 26 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
php中文本数据翻页(留言本翻页)
2006/10/09 PHP
安装ImageMagick出现error while loading shared libraries的解决方法
2014/09/23 PHP
PHP浮点数的一个常见问题
2016/03/10 PHP
PHP编程中的Session阻塞问题与解决方法分析
2017/08/07 PHP
PHP实现求两个字符串最长公共子串的方法示例
2017/11/17 PHP
Yii2压缩PHP中模板代码的输出问题
2018/08/28 PHP
php实现微信分享朋友链接功能
2019/02/18 PHP
jquery HotKeys轻松搞定键盘事件代码
2008/08/30 Javascript
IE iframe的onload方法分析小结
2010/01/07 Javascript
js换图片效果可进行定时操作
2014/06/09 Javascript
jQuery判断数组是否包含了指定的元素
2015/03/10 Javascript
关于JS变量和作用域详解
2016/07/28 Javascript
Javascript 闭包详解及实例代码
2016/11/30 Javascript
微信小程序 封装http请求实例详解
2017/01/16 Javascript
jQuery插件echarts实现的去掉X轴、Y轴和网格线效果示例【附demo源码下载】
2017/03/04 Javascript
AngularJS中$http的交互问题
2017/03/29 Javascript
Vue.js使用$.ajax和vue-resource实现OAuth的注册、登录、注销和API调用
2017/05/10 Javascript
Webstorm2016使用技巧(SVN插件使用)
2018/10/29 Javascript
D3.js(v3)+react 实现带坐标与比例尺的柱形图 (V3版本)
2019/05/09 Javascript
Django添加favicon.ico图标的示例代码
2018/08/07 Python
Django 路由控制的实现代码
2018/11/08 Python
使用python脚本自动创建pip.ini配置文件代码实例
2019/09/20 Python
tensorflow安装成功import tensorflow 出现问题
2020/04/16 Python
css3动画事件—webkitAnimationEnd与计时器time事件
2013/01/31 HTML / CSS
CSS3 文字动画效果
2020/11/12 HTML / CSS
全球航班旅行搜索网站:Cheapflights
2017/05/19 全球购物
绿色美容,有机护肤品和化妆品:Safe & Chic
2018/10/29 全球购物
node中使用shell脚本的方法步骤
2021/03/23 Javascript
银行介绍信范文
2014/01/10 职场文书
校园学雷锋活动月总结
2014/03/09 职场文书
财务管理专业求职信
2014/06/11 职场文书
夏季药店促销方案
2014/08/22 职场文书
授权委托书协议书
2014/10/16 职场文书
酒店开业主持词
2015/07/02 职场文书
如何书写公司员工保密协议?
2019/06/27 职场文书
SpringBoot2零基础到精通之数据与页面响应
2022/03/22 Java/Android