对比Python中的赋值, 浅拷贝与深拷贝

本文翻译自 Assignment, Shallow Copy, Or Deep Copy?
作者:Weilin Li
译者:耐心的农夫2020
时间:2020-03-02 14:30:00

关于Python内存管理的故事

from copy import copy, deepcopy

本文的目的是解释当我们使用下面三个python语句时内存中发生了什么。

  • 给一个变量赋值: B = A
  • 浅拷贝一个变量: C = copy(A)
  • 深拷贝一个变量: D = deepcopy(A)

首先我会简单描述一下Python的内存管理和内存优化,有了这个背景之后,我会解释一下Python中赋值语句、浅拷贝和深拷贝的差异。最后我会在一张表格中总结它们之间的差异。

如果你喜欢观看视频而不是阅读文章,你可以在此处找到补充视频。


Python的内存管理

int, float, list, dict, class instances, … 这些都是Python中的对象。在CPython实现中,内置函数id()可以返回一个对象的内存地址。

>>> L1 = [1, 2, 3]
>>> id(L1)
3061530120

如果我们创建了一个新的变量L2,这个变量指向一个值与L1相同的对象,那么L2会有一个新的内存地址。

>>> L2 = [1, 2, 3]
>>> id(L2)
3061527304

除了下面三种情况,每次创建一个新的对象,这个对象都会有一个新的内存地址。

  • 一个非常短的string
  • 在[-5, 256]区间内的整数
  • 一个空的不可改变的容器(例如 tuples)

让我们看一个整数对象的例子。变量xy都指向同一个值10,尽管在上一个例子中变量L1L2具有不同的内存地址,但是变量xy的内存地址是相同的。

>>> x = 10
>>> y = 10
>>> id(x)
2301840
>>> id(y)
2301840

这是因为,在上面提到的三种特殊情况下,Python通过让第二个变量指向了内存中同一个对象优化了内存使用。一些人把这个机制称为“共享对象”。

先记住共享对象这个概念,我们一会儿将对象深拷贝的时候还会用到它。


变量赋值

Python文档中提到

Python中的赋值语句不会拷贝对象,而只会在目标和对象之间创建绑定关系 (binding) 。

上面这句话的意思是,当我们通过赋值语句创建一个变量的时候,新的变量和原来的变量指向同一个对象。

>>> A = [1, 2, [10, 11], 3, [20, 21]]
>>> B = A
>>> id(A)
3061527080
>>> id(B)
3061527080

因为新的变量B和原始变量A共享同一个对象 (也就是同一个list),它们也包含相同的元素。

>>> id(A[2])
3061527368
>>> id(B[2])
3061527368

如下图所示,AB共享同一个id,即它们指向内存中相同的对象,并且它们也包含相同的元素。

变量赋值内存示意图

浅拷贝

当我们使用浅拷贝创建一个变量的时候,新变量指向一个新对象。

>>> A = [1, 2, [10, 11], 3, [20, 21]]
>>> C = copy(A)
>>> id(A)
3062428488
>>> id(C)
3062428520

尽管AC指向不同的对象(即内存地址不同的两个list),但两个list中每个元素都指向相同的对象。

>>> id(A[0])
2301696
>>> id(C[0])
2301696
>>> id(A[2])
3062464904
>>> id(C[2])
3062464904

下图展示了A中的元素和C中的元素是如何指向相同对象的。

浅拷贝内存示意图

深拷贝

和浅拷贝相似,当我们使用深拷贝创建一个变量的时候,新变量指向一个新对象。

>>> A = [1, 2, [10, 11], 3, [20, 21]]
>>> D = deepcopy(A)
>>> id(A)
3062727496
>>> id(D)
3062428488

Python文档 所述

浅拷贝和深拷贝之间的区别仅与复合对象(包含其他对象的对象,像lists, 或者类实例 )有关。

  • 浅拷贝会构建一个新的复合对象,并(尽可能地)将在原来复合对象中找到的对象的引用(references)插入新的复合对象。
  • 深拷贝会构建一个新的符合对象,并且递归地将在原来复合对象中找到的对象的副本(copies)插入新的复合对象。

所以与浅拷贝不同,在两个list中的元素现在指向不同的对象。

>>> id(A[0])
2301696
>>> id(D[0])
2301696
>>> id(A[2])
3062464648
>>> id(D[2])
3062466376

但是为什么A[0]D[0] 共享同一个对象(即拥有相同的内存地址)呢?因为它们都指向整数,这属于我们上面提到的内存优化三种特殊情况之一。

下图展示了AD 是如何指向内存中两个不同的lists以及 AD中的元素是如何指向不同的对象(排除因为内存优化的整数元素)。

深拷贝内存示意图

总结

下表是本文的要点。变量赋值不会拷贝对象,所以AB有相同的内存地址,并且包含相同的元素。浅拷贝为变量C创建了新的对象,但C中的元素和A中的元素仍然指向相同的对象。深拷贝为变量D创建了新的对象,并且除了三种特殊情况外,C中的元素和A中的元素都指向不同的对象。

赋值语句 vs. 浅拷贝 vs. 深拷贝

本文的灵感来自

Python documentation on copy
Understanding Python variables and Memory Management

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352