Python传值还是传引用

Python传值还是传引用

1、Python可变对象与不可变对象

不可变对象:该对象所指向的内存中的值不能被改变。当改变某个变量时候,由于其所指的值不能被改变,相当于把原来的值复制一份后再改变,这会开辟一个新的地址,变量再指向这个新的地址。
数值类型(int和float)、字符串str元组tuple都是不可变类型。

可变对象:该对象所指向的内存中的值可以被改变。变量(准确的说是引用)改变后,实际上是其所指的值直接发生改变,并没有发生复制行为,也没有开辟新的出地址,通俗点说就是原地改变。
列表list字典dict集合set是可变类型,等!(除了少数不可变对象)

2、内存示例

首先需要明确的是,Python中一切事物皆对象,变量是对对象在内存中的存储和地址的抽象。
Python传值还是传引用_第1张图片
“=”(赋值号)是将右侧对象的内存地址赋值给左侧的变量。

以下例子详细说明:
a = "abc"

Python解释器顺序干了两件事情:
1、在内存中创建一个字符串“abc”;
2、在内存中创建一个名为“a”的变量,并将“a”指向字符串“abc”(将“abc”的地址保存到“a”中)。
这样我们就能通过操作“a”而改变内存中的“abc”。

a = "123"
b = a
a = "xyz"

执行第一句Python解释器创建字符串“123”和变量“a”,并把“a”指向“123”。
在这里插入图片描述
执行第二句,因为“a”已经存在,并不会创建新的对象,但会创建变量“b”,并把“b”指向“a”指向的字符串“123“。
Python传值还是传引用_第2张图片
执行第三句,首先会创建字符串“xyz”,然后把“xyz”的地址赋予“a“(“a”指向字符串“xyz”)。
Python传值还是传引用_第3张图片
我们可以通过调用id()方法查看变量所指向对象在内存中的地址。

3、值传递还是引用传递

Python参数传递统一使用的是引用传递方式。因为Python对象分为可变对象(list,dict,set等)和不可变对象(number,string,tuple等),当传递的参数是可变对象的引用时,因为可变对象的值可以修改,因此可以通过修改参数值而修改原对象,这类似于C语言中的引用传递;当传递的参数是不可变对象的引用时,虽然传递的是引用,参数变量和原变量都指向同一内存地址,但是不可变对象无法修改,只能复制一份,所以参数的重新赋值不会影响原对象,这类似于C语言中的值传递。

4、浅拷贝(copy)和深拷贝(deepcopy)

既然Python只允许引用传递,那有没有办法可以让两个变量不再指向同一内存地址呢?
Python提供了一个copy模块,帮助我们完成这件事。

1、不使用copy模块

Python传值还是传引用_第4张图片
可以发现变量“a”,“b”指向同一块内存区域,所以对其中一个的操作将会影响到另一个。

2、使用copy模块

Python传值还是传引用_第5张图片
变量“e”和“f”是通过copy方式创建的,可以看见他们的id互不相同,并且与“c”不同,说明采用copy方式会将对象拷贝一份到新的内存地址中。
但copy和deepcopy有什么区别呢?

3、接着往下实验

Python传值还是传引用_第6张图片
从红色框出来处可以看见,虽然使用copy()方法,变量“i”指向的内存和“g”不再相同,但是“i”和“g”第二层列表还是同一个地址。但是deepcopy()方法第二层列表的地址也和“g”不同了。
所以我们说,copy()是浅拷贝,不管对象多么复杂,都只拷贝第一层。
Python传值还是传引用_第7张图片
deepcopy()是深拷贝,完全复制原变量的所有层的所有数据,在内存中生成一套完全相同的内容。
Python传值还是传引用_第8张图片

你可能感兴趣的:(编程语言)