前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >list 和 dict 的复制

list 和 dict 的复制

作者头像
不可言诉的深渊
发布2021-02-07 16:32:51
3430
发布2021-02-07 16:32:51
举报

我们都知道,Python 中有两种可变的数据类型:list 和 dict。这两种数据类型对应的实例也有很多方法可以对自身进行修改,需要注意的是,这里调用修改相关的方法的时候不是返回修改后的实例,而是就地修改,也就是原地修改。我们有些时候不希望原来的被修改,因此,复制它们的实例就显得非常重要。

元素复制(针对 list)

我们都知道,一个 list 乘上一个整数 n 表示重复 list 中的元素 n 次创建一个新 list。这里需要注意的是 n ≥ 0,如果 n < 0,返回新的空 list(我也不知道为什么,记就完事了),不相信可以看下面的示例。

代码语言:javascript
复制
>>> [0, 1] * 2[0, 1, 0, 1]>>> [0, 1] * 1[0, 1]>>> [0, 1] * 0[]>>> [0, 1] * -1[]>>> [0, 1] * -2[]

由此可以得出,创建 10 个元素全为 0 的 list 可以这么做:[0] * 10,我们来看看是不是彻底实现了复制。

代码语言:javascript
复制
>>> a = [0] * 10>>> a[0, 0, 0, 0, 0, 0, 0, 0, 0, 0]>>> a[0] = 1>>> a[1, 0, 0, 0, 0, 0, 0, 0, 0, 0]

果然实现了复制,修改第一个元素的值对其余的元素没有影响。然而,事情可能没有那么简单,我们再来看一个例子。

代码语言:javascript
复制
>>> a = [[]] * 10>>> a[[], [], [], [], [], [], [], [], [], []]>>> a[0].append(0)>>> a[[0], [0], [0], [0], [0], [0], [0], [0], [0], [0]]

我们可以发现,结果并不是我们想要的,一改全改。因此,我们发现乘法有些时候不能进行复制。与此同时,我们希望大 list 里面的小 list 也实现复制,而不是指向同一个,怎么做?很简单,使用列表推导式就可以了,还是这个例子。

代码语言:javascript
复制
>>> a = [[] for _ in range(10)]>>> a[[], [], [], [], [], [], [], [], [], []]>>> a[0].append(0)>>> a[[0], [], [], [], [], [], [], [], [], []]

这下应该没有问题了,针对 list 的元素复制到此结束。如果进行元素复制怕遇到这样的麻烦就全部使用列表推导式。但是,乘法也要会,因为有些模块或者是项目的源码可能会使用乘法!

整体复制

讲完了元素复制,我们来看一下整体复制。因为整体复制针对 list 和 dict 都可以使用,所以我分成两部分进行讲解,先看 list 的整体复制,再看 dict 的整体复制。

list 整体复制

我们假设有一个 list 的实例 a,要把它复制给 b,我们能不能直接写 b = a 呢?其实是不行的,不信的话可以看一下下面的示例。

代码语言:javascript
复制
>>> a = [1, 2, 1]>>> b = a>>> b[2] = 3>>> a[1, 2, 3]

我们可以发现,修改 b 的时候,a 也跟着改了,因此我们可以看出 a 和 b 是同一个 list,如何让 a 和 b 只是值相等,但它们是不同的 list 呢?

最容易想到的方法就是使用 list 的方法 copy,还是这个例子。

代码语言:javascript
复制
>>> a = [1, 2, 1]>>> b = a.copy()>>> b[2] = 3>>> a[1, 2, 1]

我们可以发现修改 b 对 a 没有任何影响,因此完成了复制。

完成复制的操作其核心代码就是第二行,我们还可以使用下面几种方法来完成复制。

代码语言:javascript
复制
>>> b = a[:]>>> b = list(a)>>> b = [_ for _ in a]>>> b = a + []  # 写成 b = [] + a 也可以>>> b = a * 1  # 写成 b = 1 * a 也可以

其中后面两种可能不是那么容易能够想得到的,这些方法都能完成复制,大家可以自己进行验证,我就不做演示了。

但是,这几种复制的方法都存在一个问题,我拿使用 copy 方法进行复制来演示一下这个问题。

代码语言:javascript
复制
>>> a = [[], []]>>> b = a.copy()>>> b[0].append(0)>>> a[[0], []]

我们可以发现复制并没有那么彻底,我们把这种不彻底的复制称之为浅复制。那么,如何不让它进行所谓的浅复制?我们后面再说,先继续看 dict 的整体复制。

dict 整体复制

我们假设有一个 dict 的实例 a,要把它复制给 b,我们能不能直接写 b = a 呢?其实是不行的,不信的话可以看一下下面的示例。

代码语言:javascript
复制
>>> a = {'1': 1, '2': 2}>>> b = a>>> b['1'] = 0>>> a{'1': 0, '2': 2}

我们可以发现,当 b 修改对应键的值时,a 也跟着修改,因此我们可以看出 a 和 b 是同一个 dict,如何让 a 和 b 只是值相等,但它们是不同的 dict 呢?

最容易想到的方法就是使用 dict 的方法 copy,还是这个例子。

代码语言:javascript
复制
>>> a = {'1': 1, '2': 2}>>> b = a.copy()>>> b['1'] = 0>>> a{'1': 1, '2': 2}

我们可以发现修改 b 对 a 没有任何影响,因此完成了复制。

完成复制的操作其核心代码就是第二行,我们还可以使用下面几种方法来完成复制。

代码语言:javascript
复制
>>> b = dict(a)>>> b = {k: a[k] for k in a}

但是,这几种复制的方法都存在一个问题,我拿使用 copy 方法进行复制来演示一下这个问题。

代码语言:javascript
复制
>>> a = {'1': [1], '2': [2]}>>> b = a.copy()>>> b['1'].append(1)>>> a{'1': [1, 1], '2': [2]}

我们可以发现这样的复制都是浅复制,如何不进行浅复制呢?答案很简单,进行深复制即可。

深复制

我首先来讲一下什么是深复制。深复制,即同时复制值及其包含的所有值,等等。大家看了这么专业的解释可能会觉得云里雾里,不过不用担心,我可以让大家彻底理解深复制,只要记住一句话就行:“深复制是彻底的复制!

深复制我们没有必要自己实现,Python 有模块已经给你实现好了,开箱即用就完事了。还是使用上面的例子进行演示。

代码语言:javascript
复制
>>> a = {'1': [1], '2': [2]}>>> from copy import deepcopy>>> b = deepcopy(a)>>> b['1'].append(1)>>> a{'1': [1], '2': [2]}

同样的,对于 list 套 list 的情况也必须使用深复制。

本文参与?腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2021-01-22,如有侵权请联系?cloudcommunity@tencent.com 删除

本文分享自 Python机器学习算法说书人 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与?腾讯云自媒体分享计划? ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
http://www.vxiaotou.com