一. 赋值运算

l1 = [1, 2, 'conan', [11, 22]]
l2 = l1 l1[0] = 111
print(l1) # [111, 2, 'conan', [11, 22]]
print(l2) # [111, 2, 'conan', [11, 22]]
print(id(l1[0])) # 1872924208
print(id(l2[0])) # 1872924208 l1.append(333)
print(l1) # [111, 2, 'conan', [11, 22], 333]
print(l2) # [111, 2, 'conan', [11, 22], 333]
print(id(l1)) # 2211423993544
print(id(l2)) # 2211423993544 l1[3].append(666)
print(l1) # [111, 2, 'conan', [11, 22, 666], 333]
print(l2) # [111, 2, 'conan', [11, 22, 666], 333]
print(id(l1[3])) # 2211453800136
print(id(l2[3])) # 2211453800136

​ 从上面的运行结果可以看到, 对于赋值运算来说, l1与l2指向的是同一个内存地址, 所以它们是完全一样的. 其中一个变量对列表进行改变, 剩下的变量在使用列表时, 就是使用的改变之后的列表.

二. 浅copy

# 同一代码块下
l1 = [1, 2, 'conan', [11, 22]]
l2 = l1.copy() l1[0] = 111
print(l1) # [111, 2, 'conan', [11, 22]]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1[0])) # 1872924208
print(id(l2[0])) # 1872920688 print(id(l1[1])) # 1872920720
print(id(l2[1])) # 1872920720 l1.append(333)
print(l1) # [111, 2, 'conan', [11, 22], 333]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1)) # 2343138115272
print(id(l2)) # 2343138954824 l1[3].append(666)
print(l1) # [111, 2, 'conan', [11, 22, 666], 333]
print(l2) # [1, 2, 'conan', [11, 22, 666]]
print(id(l1[3])) # 2343138954952
print(id(l2[3])) # 2343138954952
# 不同代码块下
l1 = [1, 2, 'conan', [11, 22]]
l2 = l1.copy() l1[0] = 111
print(l1) # [111, 2, 'conan', [11, 22]]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1[0])) # 1872924208
print(id(l2[0])) # 1872920688 print(id(l1[1])) # 1872920720
print(id(l2[1])) # 1872920720 l1.append(333)
print(l1) # [111, 2, 'conan', [11, 22], 333]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1)) # 2402040017736
print(id(l2)) # 2402040018120 l1[3].append(666)
print(l1) # [111, 2, 'conan', [11, 22, 666], 333]
print(l2) # [1, 2, 'conan', [11, 22, 666]]
print(id(l1[3])) # 2402040003848
print(id(l2[3])) # 2402040003848

对于浅copy来说, 只是在内存中重新开辟了一个空间存放一个新列表, 但是新列表里面的元素与原列表里面的元素的内存地址是同一个. 所以, l1与l2的id不同, 但是里面内容的id是相同的.

注意

我们发现, 在更改l1的第一个元素, 也就是把 1 改为 111 之后, 内存地址是不一样的, 这是因为int类型是不可变的数据类型; 而列表时可变的数据类型, 当里面的列表增加一个元素后, 其内存地址是不变的.

三. 深copy

依旧是先看下代码

import copy
l1 = [1, 2, 'conan', [11, 22]]
l2 = copy.deepcopy(l1) l1[0] = 111
print(l1) # [111, 2, 'conan', [11, 22]]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1[0])) # 1872924208
print(id(l2[0])) # 1872920688 print(id(l1[1])) # 1872920720
print(id(l2[1])) # 1872920720 l1.append(333)
print(l1) # [111, 2, 'conan', [11, 22], 333]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1)) # 2698658605000
print(id(l2)) # 2698660016776 l1[3].append(666)
print(l1) # [111, 2, 'conan', [11, 22, 666], 333]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1[3])) # 2698659950152
print(id(l2[3])) # 2698660016712

对于深copy来说, 深copy后的列表是在内存中重新创建的. 而且Python为了节省内存以及提高性能, 它做了这么一件事:

在深copy后的列表中,

如果其元素是可变的数据类型就再重新创建一个;

如果其元素是不可变的数据类型, 就公用一个, 也就是其内存地址是一样的.

在上面代码中的运行结果也能看出来.

四. 总结

浅copy: 嵌套的可变的数据类型 是 同一个.

深copy: 嵌套的可变的数据类型 不是 同一个.

五. 练习

5.1 练习1

写出下面代码的运行结果

l1 = [1, 2, 3, ['conan']]
l2 = l1[::]
l1[-1].append(666)
print(l1)
print(l2)

l1: [1, 2, 3, ['conan', 666]]

l2: [1, 2, 3, ['conan', 666]]

通过打印其id可以发现, l2 = l1[::] 的操作其实与浅copy一样.

『Python基础』第20节:深浅copy的更多相关文章

  1. 『Python基础』第2节: Python简介及入门

    一. Python介绍 Python是一门高级计算机程序设计语言,1989年,荷兰的Guido von Rossum创造了它.Guido是是一个牛人,1982年,他从阿姆斯特丹大学获得了数学和计算机硕 ...

  2. 『Python基础』第4节:基础数据类型初识

    本节只是对基础数据类型做个简单介绍, 详情会在之后慢慢介绍 什么是数据类型? 我们人类可以分清数字与字符串的区别, 可是计算机不能. 虽然计算机很强大, 但在某种程度上又很傻, 除非你明确告诉它数字与 ...

  3. 『Python基础』第7节:基本运算符

    一. 基本运算符 运算按种类可以分为: 算数运算.比较运算.逻辑运算.赋值运算.成员运算.身份运算.位运算. 今天我们只学习算数运算.比较运算.逻辑运算.赋值运算.成员运算 1.1 算数运算 以下假设 ...

  4. 『Python基础』第5节:条件控制

    if 语句的使用 单分支 if 条件: 满足条件后要执行的代码 例如: if 2 < 3: print(222) print(333) 每个条件后面都要使用冒号 :, 表示接下来是满足条件后要执 ...

  5. 『Python基础』第39节 函数的返回值

    1. 函数的返回值 ​ 一个函数就是封装一个功能, 这个功能一般都会有一个最终结果的. ​ 比如写一个登录的函数, 最终登录是否成功你总得告诉我一声吧? ​ 还有咱们之前也用过 len() 这个函数, ...

  6. 『Python基础』第8节:格式化输出

    现在有一个需求, 询问用户的姓名, 年龄, 工作, 爱好, 然后打印成以下格式 ************ info of Conan ************ name: Conan age: 23 ...

  7. 『Python基础』第6节:流程控制之while循环

    在生活中经常遇到循环的事情, 比如循环列表播放歌曲等. 在Python中, 也有循环, 就是其流程控制语句while. 1. 基本循环 while 条件: 循环体 # 如果条件为真, 那么就执行循环体 ...

  8. 『Python基础』第1节 Windows环境下安装Python3.x

    一. Python安装 1. 下载安装包 https://www.python.org/downloads/release/python-374/ # 3.7安装包 # 如需安装python2.7版本 ...

  9. 『Go基础』第8节 格式化输出

    输出就是将数据信息打印到电脑屏幕上. 本节我们就来学习一下Go语言中的三种输出方式: Print().Println().Printf(). 1.Print() Print()主要的一个特点就是打印数 ...

随机推荐

  1. java实现获取当前年、月、日 、小时 、分钟、 秒、 毫秒

    转载 : https://blog.csdn.net/qq_36652619/article/details/85621020 package com.app.test; import java.te ...

  2. vs2017 编译linux项目

    官方文档1: https://blogs.msdn.microsoft.com/vcblog/2017/04/11/linux-development-with-c-in-visual-studio/ ...

  3. scrapy 一些坑

    scrapy爬虫出现Forbidden by robots.txt # Obey robots.txt rulesROBOTSTXT_OBEY = False scrapy定时执行抓取任务 用cron ...

  4. 透过字节码分析Java动态代理机制。

    一.创建动态代理代码 1.创建接口 public interface Subject { void request(); } 2.创建接口实现类 public class RealSubject im ...

  5. locust性能测试脚本模板

    locust性能测试脚本模板 #!/usr/bin/env python # -*- coding: utf-8 -*- import time from locust import HttpLocu ...

  6. Javascript事件派发-dispatchEvent

    事件派发的作用: 1.派发数据,将一个封闭模块中的数据传递给另一个封闭模块.2.事件完成了较为复杂的解耦. 事件和回调函数不同在于: 1.事件可以在任意地方去获取,而回调函数只能在一个地方存在,如果需 ...

  7. C++提示没有与这些操作数匹配的<<运算符

    应该是忘了#include.#include<string>

  8. docker配置远程管理端口

    date: 2019-07-19  17:30:01 author: headsen chen notice :个人原创 Ubuntu18用这篇文章 ubuntu 16用另外一篇文章 ubuntu 1 ...

  9. 【Tomcat】Tomcat 原理架构(一)

    Tomcat是什么 开源的 Java Web 应用服务器,实现了 Java EE(Java Platform Enterprise Edition)的部 分技术规范,比如 Java Servlet.J ...

  10. [整理]Python程序员面试前需要看的博客(持续整理)

    基本素养 如何聪明的提问 面试方法 从面试官角度来告诉大家,哪些人能面试成功 如何在面试中介绍自己的项目经验 计算机系统 [面试] 迄今为止把同步/异步/阻塞/非阻塞/BIO/NIO/AIO讲的这么清 ...