python中的深拷贝和浅拷贝_python浅复制和深复制的区别

python中的深拷贝和浅拷贝_python浅复制和深复制的区别薇尔莉特来了~~~~~~

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE稳定放心使用

这一篇的内容主要是关于python中浅拷贝和深拷贝的原理。博主的其他内容可以在其他专栏中查看,更多内容还在更新中。

在这里插入图片描述
变量赋值的底层逻辑
在这里插入图片描述
在这里插入图片描述

变量赋值机制

num1 = [1,2,3]
num2 = num1 #num1将地址值复制给了num2 (参考将寄存箱小票指向的位置,复制给了num2)
num2.append(4) #所以num2修改了箱子里面的内容,num1取出来的内容也就变了。
print("num1:",num1,id(num1))
print("num2:",num2,id(num2))
#输出
''' num1: [1, 2, 3, 4] 2676183109952 num2: [1, 2, 3, 4] 2676183109952 '''

num1把地址复制给了num2。所以num1和num2指向堆内存中的同一个位置。在这种情况下,无论num1还是num2改变都会导致另一个的改变。

在这里插入图片描述

浅拷贝

再看看下面这个例子:

num1 = [10,[2,3]]
num2 = num1.copy() #将num1所指向的数据内容(地址值),复制了一份给num2

在这里插入图片描述

当执行程序后

num1 = [10,[2,3]]
num2 = num1.copy() #将num1所指向的数据内容(地址值),复制了一份给num2
num2.append(4) 
print("num1:",num1,id(num1)) #num1和num2的地址值不同
print("num2:",num2,id(num2))
num1[0] = 8 
print("num1[0]:",num1[0],id(num1[0]))
print("num2[0]:",num2[0],id(num2[0]))
print("num1[1]:",num1[1],id(num1[1]))
print("num2[1]:",num2[1],id(num2[1]))
#输出:
#num1: [10, [2, 3]] 1664063024384
#num2: [10, [2, 3], 4] 1664063016256
#num1[0]: 8 1664010119696
#num2[0]: 10 1664010119760
#num1[1]: [2, 3] 1664063016576
#num2[1]: [2, 3] 1664063016576

为什么会出现这样的输出结果?

在这里插入图片描述

我们再来看下面这个程序的输出结果

num1 = [10,[2,3]]
num2 = num1.copy() 
num2[1].append(4) 
print(num1)
print(num2)
# 输出
# [10, [2, 3, 4]]
# [10, [2, 3, 4]]

在这里插入图片描述

深拷贝

import copy
num1=[10.[2,3]]
num2=copy.deepcopy(num1)
#在使用深拷贝时,需要引用模块copy

在这里插入图片描述

我们对列表num1,num2做一定的修改

import copy
num1 = [10,[2,3]]
num2 = copy.deepcopy(num1) #将num1所指向的数据元素,复制了一份给num2
# num1[-1].append(8) #所以,修改num1的列表元素中的内容,num2中对应的列表元素也变了
print("num1中列表的地址:",id(num1[-1]),"\nnum2中列表的地址:",id(num2[-1]))
#说明列表中的”列表元素“没有复制元素内容,仅仅复制的是地址。 print("num1:",num1,id(num1))
print("num2:",num2,id(num2))
print("num1[0]:",num1[0],id(num1[0]))
print("num2[0]:",num2[0],id(num2[0]))
print("num1[1]:",num1[1],id(num1[1]))
print("num2[1]:",num2[1],id(num2[1]))
#输出
#num1中列表的地址: 1814079840768 
#num2中列表的地址: 1814079814848
#num2: [10, [2, 3]] 1814079849088
#num1[0]: 10 1814030674512
#num2[0]: 10 1814030674512
#num1[1]: [2, 3] 1814079840768
#num2[1]: [2, 3] 1814079814848

在这里插入图片描述
总结

浅拷贝:只能copy列表的一级元素,复制了嵌套的可变数据类型的地址
深拷贝:能够copy列表所有层级的元素,复制了嵌套的可变数据类型元

在这里插入图片描述

没有合适的画图工具,有些东西没有解释的很清楚,还请见谅。欢迎大家在评论区指正和交流。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/188435.html原文链接:https://javaforall.net

(0)
上一篇 2026年4月16日 下午5:10
下一篇 2026年4月16日 下午5:16


相关推荐

  • JavaSwing_2.3: JRadioButton(单选按钮)「建议收藏」

    JavaSwing_2.3: JRadioButton(单选按钮)「建议收藏」JRadioButton,单选按钮。

    2022年5月30日
    29
  • journalctl「建议收藏」

    journalctl「建议收藏」systemd提供了自己的日志系统(loggingsystem),称为journal。使用systemd日志,无需额外安装日志服务(syslog)。读取日志的命令:#journalctl默认情况下(当Storage=在文件/etc/systemd/journald.conf中被设置为auto),日志记录将被写入/var/log/journal/。该目录是sys…

    2022年5月24日
    64
  • ceph常用命令详解_ceph osd

    ceph常用命令详解_ceph osd1.OSD概念OSD:ObjectStorageDevice,主要负责响应客户端请求返回具体数据的守护进程,一般一个集群会有多个OSD,每一块盘都会对应一个OSD。2.OSD状态[root@data1~]#cephosdstat4osds:3up(since23m),3in(since13m);epoch:e345OSD状态说明:a.集群内(in)b.集群外(out)c.活着且在运行(up)d.挂了且不再运行(down).

    2025年6月29日
    4
  • URL转码

    URL转码不管是以何种方式传递 url 时 如果要传递的 url 中包含特殊字符 如想要传递一个 但是这个 会被 url 会被编码成空格 想要传递 amp 被 url 处理成分隔符 尤其是当传递的 url 是经过 Base64 加密或者 RSA 加密后的 存在特殊字符时 这里的特殊字符一旦被 url 处理 就不是原先你加密的结果了 url 特殊符号及对应的编码符号 url 中的含义编码 URL 中 号表示空格

    2026年3月20日
    2
  • web漏洞扫描工具集合「建议收藏」

    web漏洞扫描工具集合「建议收藏」最好用的开源Web漏洞扫描工具梳理链接:www.freebuf.com/articles/web/155209.html赛门铁克2017年互联网安全威胁报告中提出在他们今年扫描的网站中,有76%都

    2022年8月5日
    6
  • TRAE.ai接入Kimi-K2模型服务与国际版 Grok-4(Beta)功能升级

    TRAE.ai接入Kimi-K2模型服务与国际版 Grok-4(Beta)功能升级

    2026年3月15日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号