2-2python语法基础-基础-赋值与深浅拷贝
#####################################
预备知识一——python的变量及其存储
在详细的了解python中赋值、copy和deepcopy之前,我们还是要花一点时间来了解一下python内存中变量的存储情况。
在高级语言中,变量是对内存及其地址的抽象。
对于python而言,python的一切变量都是对象,变量的存储,采用了引用语义的方式,存储的只是一个变量的值所在的内存地址,而不是这个变量的只本身。
引用语义:在python中,变量保存的是对象(值)的引用,我们称为引用语义。采用这种方式,变量所需的存储空间大小一致,因为变量只是保存了一个引用。也被称为对象语义和指针语义。
值语义:有些语言采用的不是这种方式,它们把变量的值直接保存在变量的存储区里,这种方式被我们称为值语义,
例如C语言,采用这种存储方式,每一个变量在内存中所占的空间就要根据变量实际的大小而定,无法固定下来。
值语义和引用语义的区别:
值语义: 死的、 傻的、 简单的、 具体的、 可复制的
引用语义: 活的、 聪明的、 复杂的、 抽象的、 不可复制的
所以你就要明白了,
1,python都是引用语义,变量只是保存了一个内存地址,不是变量本身,所以占用的内存大小事固定的,
2,c语言就是值语义,保存的事整个变量,所以占据的内存大小是不固定的,
#####################################
预备知识二——各基本数据类型的地址存储及改变情况 在python中的数据类型包括:bool、int、long、float、str、set、list、tuple、dict等等。 我们可以大致将这些数据类型归类为简单数据类型和复杂的数据结构。 我的划分标准是,如果一个数据类型,可以将其他的数据类型作为自己的元素,我就认为这是一种数据结构。 数据结构的分类有很多种,但是在Python中常用的只有集合、序列和映射三种结构。对应python中的set、list(tuple、str)、dict; 常用的数据类型有int、long、float、bool、str等类型。 (其中,str类型比较特别,因为从C语言的角度来说,str其实是一个char的集合,但是这与本文的关联不大,所以我们暂时不谈这个问题) 由于python中的变量都是采用的引用语义,数据结构可以包含基础数据类型,导致每个变量中都存储了这个变量的地址,而不是值本身; 对于复杂的数据结构来说,里面的存储的也只只是每个元素的地址而已。 这也是为什么会有可变数据类型和不可变数据类型的区别!!! 可变数据类型和不可变数据类型 1.可变数据类型:在id不变的情况下,value可改变(列表和字典是可变类型,但是字典中的key值必须是不可变类型) 2.不可变数据类型:value改变,id也跟着改变。(数字,字符串,布尔类型,都是不可类型)
更新:
1.数据类型重新初始化对python语义引用的影响 变量的每一次初始化,都开辟了一个新的空间,将新内容的地址赋值给变量。我们重复的给str1赋值, 也就是
str1 = “1212” 这是一个内存地址, str1 = "2312" 在重新赋值之后就是另一个内存地址了, 2.数据结构内部元素变化重对python语义引用的影响 对于复杂的数据类型来说,改变其内部的值对于变量的影响: 当对列表中的元素进行一些增删改的操作的时候,是不会影响到lst1列表本身对于整个列表地址的,只会改变其内部元素的地址引用。 可是当我们对于一个列表重新初始化(赋值)的时候,就给lst1这个变量重新赋予了一个地址,覆盖了原本列表的地址,这个时候,lst1列表的内存id就发生了改变。 上面这个道理用在所有复杂的数据类型中都是一样的。 也就是: list1 = [12,23,34] 这是一个内存地址 list1[0] = "2323" 对列表的元素操作不会影响到列表整体的内存地址,但是列表元素的内存地址会改变, list1 = [3,3,3] 但是你对列表的整体进行赋值改变,就会影响到这个内存地址了,
#################################################
初识拷贝 我们已经详细了解了变量赋值的过程。对于复杂的数据结构来说,赋值就等于完全共享了资源,一个值的改变会完全被另一个值共享。 str2 = str1 这两个就是都是指向同一个内存地址的 list2= list1 这两个也是指向同一个内存地址的,所以是共享资源的, 然而有的时候,我们偏偏需要将一份数据的原始内容保留一份,再去处理数据,这个时候使用赋值就不够明智了。 python为这种需求提供了copy模块。提供了两种主要的copy方法,一种是普通的copy,另一种是deepcopy。我们称前者是浅拷贝,后者为深拷贝。 深浅拷贝一直是所有编程语言的重要知识点,下面我们就从内存的角度来分析一下两者的区别。
####################################################
简述Python的深浅拷贝以及应用场景? 一层的情况: import copy # 浅拷贝 li1 = [1, 2, 3] li2 = li1.copy() li1.append(4) print(li1, li2) # [1, 2, 3, 4] [1, 2, 3] # 深拷贝 li1 = [1, 2, 3] li2 = copy.deepcopy(li1) li1.append(4) print(li1, li2) # [1, 2, 3, 4] [1, 2, 3]
多层的情况: import copy # 浅拷贝 li1 = [1, 2, 3, [4, 5], 6] li2 = li1.copy() li1[3].append(7) print(li1, li2) # [1, 2, 3, [4, 5, 7], 6] [1, 2, 3, [4, 5, 7], 6] # 深拷贝 li1 = [1, 2, 3, [4, 5], 6] li2 = copy.deepcopy(li1) li1[3].append(7) print(li1, li2) # [1, 2, 3, [4, 5, 7], 6] [1, 2, 3, [4, 5], 6]
结论:就是深copy可以copy多层的,浅copy只能管一层的,
###############################################
注意这个题:
v = dict.fromkeys(['k1','k2'],[]) # {'k1': [], 'k2': []} v['k1'].append(666) print(v) # {'k1': [666], 'k2': [666]} v['k1'] = 777 print(v) # {'k1': 777, 'k2': [666]}
###########################################
##########################################
技术改变命运