Python 函数参数引用(传值/传址)/copy/deepcopy

简介:

精简版:

传值:被调函数局部变量改变不会影响主调函数局部变量

传址:被调函数局部变量改变会影响主调函数局部变量

Python参数传递方式:传递对象引用(传值和传址的混合方式),如果是数字,字符串,元组则传值;如果是列表,字典则传址;

copy使用场景:列表或字典,且内部元素为数字,字符串或元组

deepcopy使用场景:列表或字典,且内部元素包含列表或字典

完整版:

1.传值和传址的区别

传值就是传入一个参数的值,传址就是传入一个参数的地址,也就是内存的地址(相当于指针)。他们的区别是如果函数里面对传入的参数重新赋值,函数外的全局变量是否相应改变,用传值传入的参数是不会改变的,用传址传入就会改变。

1 a=1
2 def f(b):
3     b=2
4 f(a)
5 print a

例如这段代码里面,首先声明a的值为1,把a作为参数传入到函数f里面,函数f里面对b重新赋值为2,如果是传值的形式传入a的话,a的值是不会变的,依然为1,如果以传址的形式(但是这个不是程序员能决定的)传入a,a就会变成2。这个就是传值和传址的区别。

 

2. Python中的传址和传值是怎样的呢?

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

如果函数收到的是一个可变对象(比如字典或者列表)的引用,就能修改对象的原始值——相当于传址。如果函数收到的是一个不可变对象(比如数字、字符或者元组)的引用(其实也是对象地址!!!),就不能直接修改原始对象——相当于传值。

所以python的传值和传址是根据传入参数的类型来选择的

传值的参数类型:数字,字符串,元组(immutable)

传址的参数类型:列表,字典(mutable)

1 a=1
2 def f(a):
3     a+=1
4 f(a)
5 print a

这段代码里面,因为a是数字类型,所以是传值的方式,a的值并不会变,输出为1

1 a=[1]
2 def f(a):
3     a[0]+=1
4 f(a)
5 print a

这段代码里面,因为a的类型是列表,所以是传址的形式,a[0]的值会改变,输出为[2]

3. copy和deepcopy

 不止是函数里面,函数外面的引用也同样遵循这个规则:

复制代码
1 a=1
2 b=a
3 a=2
4 print a,b
5 a=[1]
6 b=a
7 a[0]=2
8 print a,b
复制代码

第一个输出为2,1,第二个输出为 [2] [2]

b=a

所以在python中,当运行上面的代码时,如果a是字典或者列表的话,程序执行的操作并不是新建一个b变量,然后a的值复制给b,而是新建一个b变量,把b的值指向a,也就是相当于在c语言里面的新建一个指向a的指针。
所以当a的值发生改变时,b的值会相应改变。

但是,当我们想新建一个与a的值相等的b变量,同时b的值与a的值没有关联时,要怎么做?这时就用到copy与deepcopy了

复制代码
 1 import copy
 2 
 3 a=[1,2,3]
 4 b=a
 5 a.append(4)
 6 print a,b
 7 
 8 a=[1,2,3]
 9 b=copy.copy(a)
10 a.append(4)
11 print a,b
复制代码

上面的输出为:

1 [1, 2, 3, 4] [1, 2, 3, 4]
2 [1, 2, 3, 4] [1, 2, 3]

这里用了copy来让b与a相等,后面如果修改了a的值,b的值并不会改变。看来copy已经可以实现我们上面的提到的需求了,那么deepcopy又有什么用?

如果我们遇到这种情况,copy就解决不了了

1 a=[1,[1,2],3]
2 b=copy.copy(a)
3 a[1].append(4)
4 print a,b

这里输出的结果为:[1, [1, 2, 4], 3] [1, [1, 2, 4], 3]  ,这样的结果明显不是我们想要的

当列表或字典参数里面的值是列表或字典时,copy并不会复制参数里面的列表或字典,这时就要用到deepcopy了

1 a=[1,[1,2],3]
2 b=copy.deepcopy(a)
3 a[1].append(4)
4 print a,b

输出的结果为:[1, [1, 2, 4], 3] [1, [1, 2], 3]

 

本文转自ZH奶酪博客园博客,原文链接:http://www.cnblogs.com/CheeseZH/p/5165283.html,如需转载请自行联系原作者

相关文章
|
2月前
|
Python
Python之函数详解
【10月更文挑战第12天】
Python之函数详解
|
2月前
|
存储 数据安全/隐私保护 索引
Python 散列类型三以及函数基础
【10月更文挑战第11天】
Python 散列类型三以及函数基础
|
29天前
|
测试技术 数据安全/隐私保护 Python
探索Python中的装饰器:简化和增强你的函数
【10月更文挑战第24天】在Python编程的海洋中,装饰器是那把可以令你的代码更简洁、更强大的魔法棒。它们不仅能够扩展函数的功能,还能保持代码的整洁性。本文将带你深入了解装饰器的概念、实现方式以及如何通过它们来提升你的代码质量。让我们一起揭开装饰器的神秘面纱,学习如何用它们来打造更加优雅和高效的代码。
|
1月前
|
弹性计算 安全 数据处理
Python高手秘籍:列表推导式与Lambda函数的高效应用
列表推导式和Lambda函数是Python中强大的工具。列表推导式允许在一行代码中生成新列表,而Lambda函数则是用于简单操作的匿名函数。通过示例展示了如何使用这些工具进行数据处理和功能实现,包括生成偶数平方、展平二维列表、按长度排序单词等。这些工具在Python编程中具有高度的灵活性和实用性。
|
2月前
|
Python
python的时间操作time-函数介绍
【10月更文挑战第19天】 python模块time的函数使用介绍和使用。
31 4
|
2月前
|
存储 Python
[oeasy]python038_ range函数_大小写字母的起止范围_start_stop
本文介绍了Python中`range`函数的使用方法及其在生成大小写字母序号范围时的应用。通过示例展示了如何利用`range`和`for`循环输出指定范围内的数字,重点讲解了小写和大写字母对应的ASCII码值范围,并解释了`range`函数的参数(start, stop)以及为何不包括stop值的原因。最后,文章留下了关于为何`range`不包含stop值的问题,留待下一次讨论。
22 1
|
2月前
|
安全 数据处理 数据安全/隐私保护
python中mod函数怎么用
通过这些实例,我们不仅掌握了Python中 `%`运算符的基础用法,还领略了它在解决实际问题中的灵活性和实用性。在诸如云计算服务提供商的技术栈中,类似的数学运算逻辑常被应用于数据处理、安全加密等关键领域,凸显了基础运算符在复杂系统中的不可或缺性。
23 0
|
7月前
|
算法 Python 容器
Python编程 - 不调用相关choose库函数,“众数“挑选器、随机挑选器 的源码编程实现
Python编程 - 不调用相关choose库函数,“众数“挑选器、随机挑选器 的源码编程实现
89 0
|
3月前
|
算法 Python
Python编程的函数—内置函数
Python编程的函数—内置函数
|
7月前
|
算法 Python
Python编程实验四:函数的使用
Python编程实验四:函数的使用
98 0