可以说Python没有赋值,只有引用。你这样相当于创建了一个引用自身的结构,所以导致了无限循环。为了理解这个问题,有个基本概念需要搞清楚。
Python没有「变量」,我们平时所说的变量其实只是「标签」,是引用。
python中,"a=b"表示的是对象a引用对象b,对象a本身没有单独分配内存空间(重要:不是复制!),它指向计算机中存储对象b的内存。因此,要想将一个对象复制为另一个对象,不能简单地用等号操作,要使用其它的方法。如序列类的对象是(列表、元组)要使用切片操作符(即':')来做复制。
在python进行像b = a这样的赋值时,只会创建一个对a的新引用,使a的引用计数加1,而不会创建新的对象:
>>> a = 'xyz' >>> import sys >>> sys.getrefcount(a) 3 >>> b = a >>> sys.getrefcount(b) 4 >>> id(a) 88292288L >>> id(b) 88292288L
这样,当引用的对象是可变对象的时候(列表,字典,可变集合等),会产生意料之外的行为:
>>> a = [1, 2, 3, 4] >>> b = a >>> b.append(5) >>> a [1, 2, 3, 4, 5]
因为a和b引用的是同一对象,改变其中一个,另外一个也会随之改变。当我们想建立一个副本而不是引用时,可以复制对象。
复制对象一般使用copy模块:
>>> a = [1, 2, 3, 4] >>> import copy >>> b = copy.copy(a) >>> b.append(5) >>> b [1, 2, 3, 4, 5] >>> a [1, 2, 3, 4]
这样就可以了,但这种复制是一种浅复制,复制的新对象中包含的是对原始对象中的项的引用,如果对象的项为可变对象,也会产生不可控行为:
>>> a = [1, [1, 2]] >>> b = copy.copy(a) >>> b[1].append(3) >>> b [1, [1, 2, 3]] >>> a [1, [1, 2, 3]]
这时候就要使用深复制了。深复制将创建一个新对象,并递归地复制它所包含的所有对象:
>>> a = [1, [1, 2]] >>> b = copy.deepcopy(a) >>> b[1].append(3) >>> b [1, [1, 2, 3]] >>> a [1, [1, 2]]
对于不可改变的对象而言(字符串,数字,元组)等,没有必要拷贝,因为它们是不可改变的,不用担心会不经意间改动了它们。拷贝操作也只会得到原对象:
>>> a = (1, 2, 3) >>> b = copy.copy(a) >>> a is b True
对于可变对象来(列表,字典,可变集合)来说,可以分别使用内置函数list(),dict(),set()来进行浅复制,速度是比使用copy模块快的。
列表也可以使用切片进行浅复制:
>>> a = [1, 2, 3, 4] >>> b = a[:] >>> a is b False >>> b [1, 2, 3, 4]
对序列数据类型(字符串,列表,元组)进行*操作时,也仅仅是复制了对象中项的引用,如果使用*创建一个多维列表:
>>> a = [1, 2, 3] >>> b = [a] >>> c = b * 3 >>> a.append(4) >>> c [[1, 2, 3, 4], [1, 2, 3, 4], [1, 2, 3, 4]]
最好是在列表推导中使用浅复制来创建多维列表,可以避免隐式的引用共享:
>>> a = [1, 2, 3] >>> c = [list(a) for i in range(3)] >>> a.append(4) >>> c [[1, 2, 3], [1, 2, 3], [1, 2, 3]]
总结
以上就是本文关于Python中对象的引用与复制代码示例的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站其他相关专题,如有不足之处,欢迎留言指出。
与原始类型相比,对象的根本区别之一是对象是“通过引用”被存储和复制的,与原始类型值相反:字符串,数字,布尔值等 —— 始终是以“整体值”的形式被复制的。 如果我们稍微看一下复制值时发生了什么,就很容易理解了。 让我们从原始类型开始,例如一个字符串。 这里我们将 message 复制到 phrase: let message = "Hello!"; let phrase = message; 结果我
问题内容: 我正在上最后一堂课。我复制的阵列使用对象,并且当我改变一个中复制的数组对象,它就会被反映原始数组英寸 我正在输出, 我应该在所有情况下都得到25和26的输出,对吗?为什么会改变? 问题答案: System.arrayCopy()复制对象还是对对象的引用? 参考,这是一个 浅表 副本。出乎意料的是,文档没有明确地说,只是隐式地说,因为他们 只 谈论复制数组元素,而不是递归地复制他们引用的
本文向大家介绍python复制与引用用法分析,包括了python复制与引用用法分析的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了python复制与引用用法。分享给大家供大家参考。具体分析如下: 简单复制是引用 使用copy.copy进行浅拷贝 使用copy.deepcopy进行迭代拷贝,之后就可以更改新对象的属性而不影响原对象了,只是效率会下降和内存占有量会上升。 对于list,dict
我有以下C#类: 总而言之,区域设置有区域、按钮和字段。区域有中心和东部。中心和东部拥有产权。Fields包含具有属性firstName、lastName和ChooseLocale的标签。 在一个名为GetLocale的方法中,我有以下代码: 运行代码时,在以下行抛出一个“NullReferenceException was unhandled by user code”: 我设置属性title、
本文向大家介绍Python中字符串与编码示例代码,包括了Python中字符串与编码示例代码的使用技巧和注意事项,需要的朋友参考一下 在最新的Python 3版本中,字符串是以Unicode编码的,即Python的字符串支持多语言 编码和解码 字符串在内存中以Unicode表示,在操作字符串时,经常需要str和bytes互相转换 如果在网络上传输或保存到磁盘上,则从内存读到的数据就是st
问题内容: 这个问题已经在这里有了答案 : 探索和反编译python字节码[关闭] (6个答案) 7年前关闭。 假设我们有一个python字符串(不是文件,字符串,没有文件) 好?现在我们将字符串编译成一段python字节码 现在的问题是:假设我不知道TheString(代表原始字符串对象的字符串),我怎么能从Binary获得? 很快:与compile()相反的函数是什么? 问题答案: 没有源代码