第022天 内存管理和拷贝

多继承

python中的类支持多继承(让一个类同时继承多个类);
多继承的时候,子类只能继承第一个父类所有的属性和方法,后面的父类中只有字段和方法可以被继承

class Animal(object):
    num = 100

    def __init__(self):
        self.age = 0
        self.gender = '雌'

    @classmethod
    def func1(cls):
        print('动物类的类方法')

    def func2(self):
        print('动物类中的对象方法')


class Fly(object):
    name = '飞行器'

    def __init__(self):
        self.height = 100
        self.time = 5
        self.speed = 100

    def func2(self):
        print('飞行的对象方法')


class Bird(Animal, Fly):
    pass


bird1 = Bird()
# 字段都能继承
print(Bird.num, Bird.name)

Bird.func1()
bird1.func2()

# print(bird1.age, bird1.gender)
# print(bird1.speed, bird1.height, bird1.time)
运算符

python中所有的类型都是类,所以所有的数据都是对象;
python中使用任意的运算符都是在调用相应类中的相应方法,每一个运算符对应什么方法是固定的,
某种数据是否支持某个运算符操作就看这个数据类型中是否实现了对应的方法

运算符重载

指的是在不同的类中实现同样的运算符对应的函数
类的对象默认情况下只支持: ==, !=

impot copy
10 * 20
'abc'+'123'
[1, 2] + [2, 3, 4]
# {'a': 10} + {'b': 20}


class Student:
    def __init__(self, name, age, score=0):
        self.name = name
        self.age = age
        self.score = score

    def __repr__(self):
        return '<%s, id:%s>' % (str(self.__dict__)[1:-1], hex(id(self)))

    # a+b ->  a.__add__(b)
    # self -> 当前类的对象,也是+前面的那个数据
    # other -> +后面的那个数据, 类型根据运算规则的设计可以是任何类型的数据
    def __add__(self, other):
        # return self.age + other.age
        return self.score + other.score
        # return Student(self.name+other.name, self.age+other.age, self.score + other.score)
        # return self.score + other

    # a*b -> a.__mul__(b)
    def __mul__(self, other):
        list = []
        for _ in range(other):
            list.append(copy.copy(self))
        return list

    # a<b  -> a.__lt__(b)
    # 注意: <和>符号只需要重载其中一个就可以
    def __lt__(self, other):
        return self.score < other.score


stu1 = Student('小明', 19, 90)
stu2 = Student('小花', 20, 78)

# stu1.__add__(stu2)
print(stu1 + stu2)

# stu1.__add__(100)
# print(stu1 + 100)

print(stu1 * 3)

students = [stu1, stu2, Student('小红', 12, 100)]
students.sort()
print(students)

print(stu1 < stu2)
print(stu1 > stu2)
from copy import copy, deepcopy


class Dog:
    def __init__(self, name, color):
        self.name = name
        self.color = color

    def __repr__(self):
        return '<%s,id: %s>' % (str(self.__dict__)[1:-1], hex(id(self)))


class Person:
    def __init__(self, name, age, dog=None):
        self.name = name
        self.age = age
        self.dog = dog

    def __repr__(self):
        return '<%s,id: %s>' % (str(self.__dict__)[1:-1], hex(id(self)))


p1 = Person('小明', 18, Dog('大黄', '黄色'))
直接赋值

将变量中的地址直接赋给新的变量;赋值后两个变量的地址相同

p2 = p1
print(id(p1), id(p2))    # 4537270848 4537270848
p1.name = '小花'
print(p1.name, p2.name)  # 小花 小花
p2.dog.color = '绿色'
print(p1.dog.color, p2.dog.color)   # 绿色 绿色
拷贝

不管是浅拷贝还是深拷贝都会对原数据进行赋值产生新的地址

list1 = [1, 2, 3]
list2 = copy(list1)
list3 = deepcopy(list1)
print(id(list1), id(list2), id(list3))

list1.append(100)
print(list2, list3)

p3 = copy(p1)
p4 = deepcopy(p1)
print(id(p1), id(p3), id(p4))
p1.name = '小红'
print(p3.name, p4.name)
浅拷贝

字符串、列表和元祖的切片;对象.copy(); copy模块中的copy方法都是浅拷贝
浅拷贝指拷贝当前对象,不会对子对象进行拷贝

print('==========浅拷贝=========')
p3 = copy(p1)
print(id(p1), id(p3))   # 4401914600 4401916336
print(id(p1.dog), id(p3.dog))  # 4567065152 4567065152
p1.name = 'Tom'
print(p1.name, p3.name)   # Tom 小红
p1.dog.color = '红色'
print(p1.dog.color, p3.dog.color)   # 红色 红色
深拷贝

copy模块中的deepcopy方法是深拷贝

print('==========深拷贝=========')
p4 = deepcopy(p1)
print(id(p1), id(p4))   # 4486472592 4486474384
print(id(p1.dog), id(p4.dog))  # 4485063960 4486474552
p1.name = 'Bob'
print(p1.name, p4.name)   # Bob Tom
p1.dog.color = '橙色'
print(p1.dog.color, p4.dog.color)   # 橙色 红色
练习:
a = ['color', 'height', 'background']
b = [a, 'aaa', 'bbb']
c1 = b
c2 = copy(b)
c3 = deepcopy(b)
a[-1] = ['BG']
b.append('ccc')

# b = [['color', 'height', 'background'], 'aaa', 'bbb']
# 问题: print(c1), print(c2), print(c3)的结果分别是
# c1 = [['color', 'height', ['BG']], 'aaa', 'bbb', 'ccc']
# c2 = [['color', 'height', ['BG']], 'aaa', 'bbb']
# c3 = [['color', 'height', 'background'], 'aaa', 'bbb']
print(c1)
print(c2)
print(c3)

枚举值的特点:
1.可以通过有意义的属性名直接显示数据
2.每个数据的值不能修改
3.可以做到不同的数据的值唯一

from enum import Enum, unique
@unique
class PokerNum(Enum):
    J = 11
    Q = 12
    K = 13
    A = 1


class Color:
    RED = (255, 0, 0)


print(PokerNum.J)
print(PokerNum.K.value, PokerNum.J.value > PokerNum.Q.value)
内存的开辟

内存区间分为栈区间和堆区间;栈区间的内存自动开辟自动释放,堆区间的内存需要程序员手动开辟,手动释放;
但是python已经将堆区间内存的开辟和释放自动化

a.当每次给变量赋值的时候,系统会先在堆区间中开辟空间将数据存起来,然后再将数据在堆中的地址存到变量中,变量存在栈区间;
b.数字和字符串数据在开辟空间的时候,会先检测内存中之前是否已经有这个数据,
如果有就直接使用之前的数据,没有才开辟新的空间保存数据

from copy import copy, deepcopy
from sys import getrefcount
a = [1, 2, 3, [1, 2]]     # [1, 2, 0xccc]
b = [1, 2, 3, [1, 2]]     # [1, 2, 0xccc]
print(id(a), id(b))    # 4484287176 4484285768
print(id(a[3]), id(b[3]))
print(id(a[0]), id(b[0]), id(a[3][0]), id(b[3][0]))   # 4449482688 4449482688 4449482688 4449482688

a1 = 100
b1 = 100
print(id(a1), id(b1))   # 4480746528 4480746528

a2 = 'hello'
b2 = 'hello'
print(id(a2), id(b2))    # 4492420184 4492420184

a3 = {'a': 10}
b3 = {'a': 10}
print(a3 == b3)    # True
print(a3 is b3)    # False


a4 = 200
b4 = deepcopy(a4)
print(id(a4), id(b4))    # 4367007904 4367007904

a5 = '安徽省打发卡机阿哈萨克阿按时发阿萨德回复杰卡斯发是打飞机啊山东矿机返回撒娇快递费哈萨克京东方撒地方'
b5 = '安徽省打发卡机阿哈萨克阿按时发阿萨德回复杰卡斯发是打飞机啊山东矿机返回撒娇快递费哈萨克京东方撒地方'
print(id(a5), id(b5))   # 4338364064 4338364064
内存的释放

栈区间:全局栈区间在程序结束后销毁,函数栈区间在函数调用结束后销毁(自动)
堆区间: 看一个对象是否销毁,就看这个对象的引用计数是否为0;如果一个对象的引用为0,这个对象就会销毁;(垃圾回收机制)
注意: python中针对对象的循环引用已经做了处理,程序员不需要写额外的代码来解决循环引用问题

from copy import copy, deepcopy
from sys import getrefcount
a6 = {'name': '小明', 'age': 18}
b6 = a6
print(getrefcount(a6))
b6 = 100
print(getrefcount(a6))
del a6
print(getrefcount(b6))


class Person:
    def __init__(self, name):
        self.name = name
        self.dog = None


class Dog:
    def __init__(self, name):
        self.name = name
        self.owner = None


dog1 = Dog('大黄')
p1 = Person('小明')
p1.dog = dog1
dog1.owner = p1

del dog1
del p1
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,616评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,020评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,078评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,040评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,154评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,265评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,298评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,072评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,491评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,795评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,970评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,654评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,272评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,985评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,815评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,852评论 2 351

推荐阅读更多精彩内容