python中变量在内存中的存储与地址关系解析、浅度/深度copy、值传递、引用传递

时间:2021-10-19 10:39:39

---恢复内容开始---

1.变量、地址

变量的实现方式有:引用语义、值语义

  python语言中变量的实现方式就是引用语义,在变量里面保存的是值(对象)的引用(值所在处内存空间的地址)。采用这种方式,变量所需的存储空间大小一致,因为其中只需要保存一个引用。而有些语言(例如c)采用的不是这种方式,它们把变量直接保存在变量的存储区里,这种方式就称为值语义。这样的话,一个整数类型的变量就需要保存一个整数所需要的空间(例如c语言中int类型占用4个字节大小,所能表示的数的最大值为2^32,2147483647)。

  python中变量与对象的引用关系类似于c语言的指针变量与指针指向值的关系。

  在python的数据结构中,对象分为可变对象不可变对象。基本数据类型如int、float等都是不可变对象。在结构数据类型中,元祖tuple、str是不可变对象,list(列表)、dict(字典)、set(集合)是可变对象,可变对象存储的元素的引用其实是没有改变的,改变的是其引用指向的值。

  采用引用语义存储的只是一个变量的值所在的内存地址,而不是这个变量的值本身。

python中变量在内存中的存储与地址关系解析、浅度/深度copy、值传递、引用传递

如上所示,变量中存储的是值的引用,也就是指所在内存空间的地址。

python中变量在内存中的存储与地址关系解析、浅度/深度copy、值传递、引用传递

 id函数(python的内置函数,用来查看对象的身份,也就是内存地址)

  对于给变量赋值时,每一次的赋值都会产生一个新的地址空间,将新内容的地址赋值给变量,但是对于相同的值,地址不发生变化。

python中变量在内存中的存储与地址关系解析、浅度/深度copy、值传递、引用传递

  对复杂的数据类型(列表、元祖、字典),如果添加某一项元素,或者添加几个元素,不会改变其本身的地址,只会改变其内部元素的地址引用,但是如果对其重新赋值时,就会重新赋予地址覆盖就地址,这时地址就会发生改变。

python中变量在内存中的存储与地址关系解析、浅度/深度copy、值传递、引用传递

python中变量在内存中的存储与地址关系解析、浅度/深度copy、值传递、引用传递

  python中相同的值的不同对象,相当于内存中对于相同值的对象保存了对份。但是对于不可变数据类型,内存中只能有一个相同值的对象。同时要看是否产生新的对象。

2.浅拷贝和深拷贝

copy.copy():浅拷贝,不管多复杂的数据结构,浅拷贝都只会copy一层。

copy.deepcopy():深拷贝,会完全复制原变量相关的所有数据,到最后一层(自身包含的所有子列表)。在内存那种生成一套完全一样的内容。

python中变量在内存中的存储与地址关系解析、浅度/深度copy、值传递、引用传递

  由上图可以看出,浅拷贝,只拷贝一层,因此当list_重新赋值以后,浅拷贝后list_2包含的子列表发生了变化,而而深拷贝以后的列表list_3所包含的字列表并没有发生改变。深拷贝等于完全复制并重新开辟新的内存空间,和原列表两者之间互不影响。

3.值传递与引用传递:

  可变对象为引用传递,不可变对象为值传递。(函数传值)

值传递:被调函数在执行时,首先对收到的参数对象生成一个副本,在执行过程中,是对参数副本的操作,并不会对原参数产生改变。也就是在堆栈中开辟内存空间存放由主调函数传进来的实参对应的副本值。特点:函数对收到的参数的任何操作,不会对原参数(实参变量)产生影响。

python中变量在内存中的存储与地址关系解析、浅度/深度copy、值传递、引用传递

引用传递:当传递列表或者字典时,如果改变引用的值,就改变了原始对象。(引用传递直接传的是地址,是对原始对象的直接操作。)

python中变量在内存中的存储与地址关系解析、浅度/深度copy、值传递、引用传递

由上图可以看出,引用传递,函数修改的直接是实参的值。但是,在函数体中不能直接修改整个列表或者字典的值,这样做,也等于创建实参的副本,并不会对实参本身产生影响。如下图所示。

python中变量在内存中的存储与地址关系解析、浅度/深度copy、值传递、引用传递

   

  

---恢复内容结束---