验证python中函数传参是引用传递

定义:

  • 值传递(pass by value)是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数。
  • 引用传递(pass by reference)是指在调用函数时将实际参数的地址直接传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数。

从上面定义可以看出,主要区别就在于内存地址的变化。值传递,内存地址发生变化。引用传递,内存地址不发生变化。

使用User来验证

 1 def printid(user):
 2     user.age=20
 3     print("函数中user内存地址:",id(user))#函数中user内存地址: 2254536402504
 4     print("函数中user内容:",user.__dict__)#函数中user内容: {‘age‘: 20}
 5
 6 u=User()
 7 u.age=30
 8 print("u对象内容:",u.__dict__)#u对象内容: {‘age‘: 30}
 9 print("u内存地址:",id(u))#u内存地址: 2254536402504
10 printid(u)
11 print("u对象内容:",u.__dict__)#u对象内容: {‘age‘: 20}

从上面例子可以看到,内存地址都是‘2254536402504’,当age发生变化,User对象发生变化。

所以从上面的例子来看,Python是引用传递。(Java是值传递)

Python中有下面两种类型划分

可变类型(mutable):列表,字典

不可变类型(unmutable):数字,字符串,元组

如果是不可变类型,在对对象本身操作的时候,必须在内存中新申请一块区域(因为老区域#不可变#)。

如果是可变类型,对对象操作的时候,不需要再在其他地方申请内存,只需要在此对象后面连续申请(+/-)即可,也就是它的address会保持不变,但区域会变长或者变短。

下面使用数字在验证一下,Python是引用传递

 1 def printv(i,j):
 2     i=6
 3     print("i的内存地址",id(i))#i的内存地址 140725476422048
 4     print("i的值",i)#i的值 6
 5     print("j的内存地址",id(j))#j的内存地址 2908009218864
 6     print("j的值",j)#j的值 5000
 7
 8 n=5
 9 m=5000
10 print("n的内存地址",id(n))#n的内存地址 140725476422016
11 print("m的内存地址",id(m))#m的内存地址 2908009218864
12 print("n的值",n)#n的值 5
13 print("m的值",m)#m的值 5000
14 printv(n,m)
15 print("n的值",n)#n的值 5
16 print("m的值",m)#m的值 5000

一眼可以看出,m和j的内存地址是一样的,都是‘2908009218864’。

但是n和i的内存地址不一样,且最后n的值没有发生变化。

原因:数字是不可变类型,i被重新赋值,实际就是申请了一个新内存区域,已经不是原来的n。如果i没有被赋值,就会像j一样,内存地址保持不变。

结论:Python是引用传递

参考:

https://www.cnblogs.com/huamingao/p/5809936.html

https://www.cnblogs.com/yxtz271828/p/8988176.html

https://www.cnblogs.com/wchxj/p/8729503.html

原文地址:https://www.cnblogs.com/moonpool/p/Python.html

时间: 2024-08-29 05:52:37

验证python中函数传参是引用传递的相关文章

Python 中函数传参是传值还是传引用

直接简单的例子: 1 from ctypes import * 2 import os.path 3 import sys 4 5 def test(c): 6 print "test before " 7 print id(c) 8 c+=2 9 print "test after +" 10 print id(c) 11 return c 12 13 def printIt(t): 14 for i in range(len(t)): 15 print t[i]

python的函数传参补充

形参可以传递变量,列表,字典,元组 def model(name) 形参 model('zhuzai') 传递实参 def model(name,age) 位置形参 model('zhuzai',0) 传递实参 def model(name='zhuzai') 关键字形参 model(name='zhuzai2')或model() 若不传递实参,则默认为'zhuzai' def model(*name) 任意数量形参,全部实参会用一个元组封装起来,所以函数只接受了一个值 mdoel('zhuza

Javascript 之《函数传参到底是值传递还是引用传递》

前言 这个问题其实困惑了我好久,但是在实际使用中总是得过且过,不想去深究.由于这种态度,在学习 Javascript 过程中,水平一直都是出于半桶水状态,很多概念和原理似懂非懂,模糊不清. 所以,写了一系列的<Javascript 之 ...>就是为了端正态度,认真地研究一下 Javascript 的特性和原理,夯实基础. 今天,这一篇探究的是函数传参的问题:函数传参到底是传值还是传的引用? 1.如果是引用传递 var name = 'JS'; function changeName(name

python中给函数传参是传值还是传引用

首先还是应该科普下函数参数传递机制,传值和传引用是什么意思? 函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题.基本的参数传递机制有两种:值传递和引用传递. 值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本.值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值. 引用传递(pass-

python函数传参是传值还是传引用?

首先还是应该科普下函数参数传递机制,传值和传引用是什么意思? 函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题.基本的参数传递机制有两种:值传递和引用传递. 值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本.值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值. 引用传递(pass-

python中函数传值与传引用

python中函数整数,字符串,元组都是传值,函数中不会改变其值,其他的会在函数中改变其值 例如传列表: #-*-ecoding:UTF-8 -*- def fun(a):     a[0]="sss";     print a     print id(a)     return ; a=["a","b"]; fun(a); print a; print id(a) 列表中的值已经改变,但列表的地址没有改变

关于函数传参--传指针,传引用

今天和同学讨论到指针和引用的传递问题,有些想法从推理上讲是正确的,但是因为是推理,说出自己观点的时候不是那么有底气,本着实践是检验真理的唯一标准的原则,在电脑上敲了几段代码,验证了推理的正确性. 先上代码,再分析. 代码1: void Swap0(int a1,int b1){ int temp; temp=a1; a1=b1; b1=temp; } void Swap1(int *a1,int *b1){ //交换地址 int *temp; temp=a1; a1=b1; b1=a1; } v

[转]Python中函数的值传递和引用传递

首先还是应该科普下函数参数传递机制,传值和传引用是什么意思? 函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题.基本的参数传递机制有两种:值传递和引用传递. 值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本.值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值. 引用传递(pass-

python中函数参数的引用方式

值传递和引用传递时C++中的概念,在python中函数参数的传递是变量指向的对象的物理内存地址!!! python不允许程序员选择采用传值还是传引用.Python参数传递采用的肯定是"传对象引用"的方式.这种方式相当于传值和传引用的一种综合.如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值--相当于通过"传引用"来传递对象.如果函数收到的是一个不可变对象(比如数字.字符或者元组)的引用,就不能直接修改原始对象--相当于通过"传值