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)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Nginx负载均衡策略_nginx高可用集群和负载均衡集群

    Nginx负载均衡策略_nginx高可用集群和负载均衡集群nginx的负载均衡策略有4种:轮询(默认)最基本的配置方法,它是upstream的默认策略,每个请求会按时间顺序逐一分配到不同的后端服务器。参数有:项目Valuefail_timeout与max_fails结合使用max_fails设置在fail_timeout参数设置的时间内最大失败次数,如果在这个时间内,所有针对该服务器的请求都失败了,那么认为该服务器会被认为是停机了fail_time服务器会被认为停机的时间长度,默认为10s。backup标记该服

    2022年10月13日
    1
  • Python抓取数据_python抓取游戏数据

    Python抓取数据_python抓取游戏数据前言本文整理自慕课网"《Python开发简单爬虫》",将会记录爬取百度百科“python”词条相关页面的整个过程。抓取策略确定目标:确定抓取哪个网站的哪些页面的哪部分数据

    2022年8月2日
    6
  • 集成灶功能图标解释_为什么不建议买集成灶

    集成灶功能图标解释_为什么不建议买集成灶GitLab Auto DevOps功能与Kubernetes集成教程

    2022年4月22日
    67
  • WPF Visifire图表控件使用基础

    WPF Visifire图表控件使用基础https://www.cnblogs.com/wyuan/archive/2012/07/22/WPF.html引言:  由于项目中需要使用Visifire所以自己就写了一些demo,大家一起共享!基础Visifire图表的展示1.Visifire的创建需要引用的DLL包【WPFToolkit.dll;WPFVisifire.Charts;WPFVisifire.Gauges(这个以后会用到)】2

    2022年7月21日
    11
  • 传真服务器的定义与选购

    传真服务器的定义与选购 AOFAX热点:传真服务器的定义与选购服务信息卡:联系人:朱成军服务热线:15920885998 商务QQ:909075483.一、传真服务器和传真系统传真服务器定义:所谓传真服务器,就是为企业内部所有员工提供电脑传真收发服务的设备,能够长期脱离其他电脑独立工作,是传真服务器的必要条件。传真服务器由传真软件、传真卡、工控机等组成。传真系统:传真服务器在某些场合也

    2022年6月28日
    24
  • 五大常用算法之五:分支限界法

    一、基本描述类似于回溯法,也是一种在问题的解空间树T上搜索问题解的算法。但在一般情况下,分支限界法与回溯法的求解目标不同。回溯法的求解目标是找出T中满足约束条件的所有解,而分支限界法的求解目标则是找

    2021年12月25日
    33

发表回复

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

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