day18-内存管理和拷贝

1.内存管理

1)内存管理分为:开辟内存和释放内存

a.内存的开辟

内存分为栈区间和堆区间,python所有的数据都是保存在堆区间的,栈区间中保存存储数据地址的变量;
每次需要一个数据的时候,python自动去开辟相应的空间;原则上只要需要数据就会开辟一个新的空间,
哪怕数据和之前的数据是相等的也会重新开辟;但是数字和字符串特殊,每次给数字或者字符串开辟空间前
会先检查这个数据在内存是否已经存在,如果存在就直接使用之前存储的数据,不存在才会重新开辟空间。

b.内存的释放

python中内存释放采用的是垃圾回收机制
当数据(堆中)的引用计数为0的时候,数据就会被自动销毁(被回收)

引用:保存对象的地址的数据就是这个对象的引用
增加引用计数:让更多的时候存储数据的地址
减少引用:删除引用;让引用保存别的数据的地址

注意:python中针对数据的循环引用已经做了特殊处理,程序员不用再写额外的代码去处理循环应用

from sys import getrefcount
class Person:
    def __init__(self, name, age, gender):
        self.name = name
        self.age = age
        self.gender = gender
a = [1, 2]
b = [1, 2]
print(a == b)
print(id(a), id(b))   # 地址不一样

a1 = 2
b1 = 2
print(id(a1), id(b1))   # 地址一样

a2 = 'a12'
b2 = 'a12'
print(id(a2), id(b2))   # 地址一样

p1 = Person('aa', 18, '男')
p2 = Person('aa', 18, '男')
print(id(p1), id(p2))   # 地址不一样

print('===============内存释放==================')
p = Person('qq', 22, '女')  # 1
print(getrefcount(p))     # 2
p1 = p  # 2
print(getrefcount(p))   # 3
list1 = [p1, 12, 2]   # 3
print(getrefcount(p))  # 4

2.浅拷贝和深拷贝

面试题:说说python中,直接赋值、浅拷贝和深拷贝之间的区别

1)直接赋值

用一个变量给另外一个变量赋值,赋的是地址,赋值后数据只有一个,两个变量相互影响

a = [1, 2, 3]
b = a
a.append(12)
print(b)

2)浅拷贝

列表切片、列表.copy、字典.copy等都属于浅拷贝
copy模块中的copy也是浅拷贝,可以拷贝任何数据

import copy
class Person:
    def __init__(self, name, age, gender):
        self.age = age
        self.name = name
        self.gender = gender

    def __repr__(self):
        return str(self.__dict__)
a = [1, 2, 3]
b = a.copy()
a.append(10)
print(a, b)

p1 = Person('aa', 18, '男')
p2 = copy.copy(p1)
p1.name = 'bb'
print(p1, p2)

3)深拷贝

copy模块中的deepcopy是深拷贝

p1 = Person('aa', 18, '男')
p2 = copy.deepcopy(p1)
p2.name = 'cc'
print(p1, p2)

4)浅拷贝和深拷贝的区别

不管是浅拷贝还是深拷贝,赋值的时候都是先复制原数据产生一个新的数据,然后将新的地址赋给变量。
如果拷贝对象中又有子对象(除了数字和字符串以外的数据),浅拷贝地址原样复制,不会拷贝子对象;
如果深拷贝,子对象也会被拷贝

a = [100, 200]
list1 = [1, 2, a]
list2 = copy.copy(list1)
list3 = copy.deepcopy(list1)
list1[1] = 222
print(list2, list3)
a.append('1000')
print(list2, list3)   # [1, 2, [100, 200, '1000']] [1, 2, [100, 200]]

3.包的使用

1)什么是包

python中包就是拥有init.py文件的文件夹

2)怎么使用包的内容

a.import 包名 - 需要在init.py中通过‘包名.模块名’的方式导入包中相应的模块

# import test
# print(test.test1.a1)

b.import 包.模块
import 包.模块 as 新名

# import test.test1
# test.test1.func1()

# import test.test1 as test1
# test1.func1()

c.from 包 import 模块1,模块2,...
d.from 包.模块 import 变量1,变量2,...

# from test import test1
# test1.func1()

# from test.test1 import func1
# func1()
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,156评论 1 32
  • 内存管理和拷贝 review 1.内置类属性: name、doc、class、dict、module、bases2...
    漫磋嗟阅读 176评论 0 0
  • 一、Python简介和环境搭建以及pip的安装 4课时实验课主要内容 【Python简介】: Python 是一个...
    _小老虎_阅读 5,848评论 0 10
  • IO密集型程序、深拷贝和浅拷贝、模块导入、with 语句 1.1 GIL 学习目标 1. 能够说出 GIL 是什...
    Cestine阅读 900评论 0 0
  • 个人笔记,方便自己查阅使用 Py.LangSpec.Contents Refs Built-in Closure ...
    freenik阅读 67,787评论 0 5