内存管理和拷贝_笔记

一、review

1.写一个扑克类, 要求拥有发牌和洗牌的功能(具体的属性和其他功能自己根据实际情况发挥)

from enum import Enum, unique
from random import shuffle
@unique
class PokerNum(Enum):
    J = 11, 'J', 11
    Q = 12, 'Q', 12
    K = 13, 'K', 13
    A = 1, 'A', 14
    TWO = 2, '2', 15
    BigJoker = 15, '大王', 17
    SmallJoker = 14, '小王', 16


@unique
class PokerColor(Enum):
    Club = '♣'
    Diamond = '♦'
    Heart = '♥'
    Spade = '♠'
    Space = ''


class Poker:
    def __init__(self, num: PokerNum, color: PokerColor):
        self.num = num
        self.color = color

    def __repr__(self):
        if isinstance(self.num, PokerNum):
            return str(self.color.value) + str(self.num.value[1])
        return str(self.color.value)+str(self.num)

    def __gt__(self, other):
        if isinstance(self.num, PokerNum):
            a = self.num.value[2]
        else:
            a = self.num

        if isinstance(other.num, PokerNum):
            b = other.num.value[2]
        else:
            b = other.num

        return a > b


class Game:
    # 斗地主
    def __init__(self):
        pokers = []
        colors = [PokerColor.Club, PokerColor.Diamond, PokerColor.Heart, PokerColor.Spade]
        nums = [PokerNum.A, PokerNum.TWO, 3, 4, 5, 6, 7, 8, 9, 10, PokerNum.J, PokerNum.Q, PokerNum.K]
        for num in nums:
            for color in colors:
                p = Poker(num, color)
                pokers.append(p)
        pokers.extend([Poker(PokerNum.SmallJoker, PokerColor.Space), Poker(PokerNum.BigJoker, PokerColor.Space)])
        self.pokers = pokers
        self.pokers_iter = iter(self.pokers)

    # 洗牌
    def shuffling(self):
        shuffle(self.pokers)
        self.pokers_iter = iter(self.pokers)

    # 斗地主发牌方式
    def deal1(self):
        ps1 = []
        ps2 = []
        ps3 = []
        for _ in range(17):
            ps1.append(next(self.pokers_iter))
            ps2.append(next(self.pokers_iter))
            ps3.append(next(self.pokers_iter))
        ps1.sort(reverse=True)
        ps2.sort(reverse=True)
        ps3.sort(reverse=True)
        return ps1, ps2, ps3, list(self.pokers_iter)


game1 = Game()
game1.shuffling()
p1, p2, p3, di = game1.deal1()
print(p1)
print(p2)
print(p3)
print('底牌:', di)

2、(尝试)写一个类,其功能是:1.解析指定的歌词文件的内容 2.按时间显示歌词

提示:歌词文件的内容一般是按下面的格式进行存储的。
歌词前面对应的是时间,在对应的时间点可以显示对应的歌词

class Lyric:
    def __init__(self, time, word):
        value = float(time[1:3])*60 + float(time[4:])
        self.time = value
        self.word = word

    def __repr__(self):
        return '<'+str(self.time) + ':' + self.word+'>'

    def __gt__(self, other):
        return self.time > other.time


class LyricAnalysis:

    def __init__(self, name):
        # 歌名
        self.name = name
        self.__all_lyrics = []

    def get_word(self, time):
        # =======解析歌词文件======
        if not self.__all_lyrics:
            print('解析歌词')
            with open('files/'+self.name) as f:
                while True:
                    line = f.readline()
                    if not line:
                        break
                    lines = line.split(']')
                    word = lines[-1]
                    for t in lines[:-1]:
                        lyric = Lyric(t, word)
                        self.__all_lyrics.append(lyric)

            # 排序
            self.__all_lyrics.sort(reverse=True)

        # ==========获取歌词==========
        for lyric in self.__all_lyrics:
            if lyric.time <= time:
                return lyric.word


ly = LyricAnalysis('蓝莲花')
print('===:',ly.get_word(123))
print('===:',ly.get_word(10))
print('===:',ly.get_word(89))

ly2 = LyricAnalysis('一首简单的歌')
print('!!!:', ly2.get_word(30))
print('!!!:', ly2.get_word(90))

二、多继承

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)

三、运算符重载

1.运算符

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

2、运算符重载

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

import copy
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)

四、拷贝

1、直接赋值

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

2、浅拷贝

拷贝(不管是浅拷贝还是深拷贝)不同于直接赋值,会产生新的地址,
字符串、列表、元祖的切片,对象.copy(),copy模块中的copy方法都是浅拷贝
浅拷贝只会拷贝当前对象,不会对子对象进行拷贝

3、深拷贝

copy模块中的deepcopy方法是深拷贝,不仅会拷贝当前对象,还会拷贝子对象,产生新的地址

# 练习:
import copy
a = ['color', 'height', 'background']
b = [a, 'aaa', 'bbb']
c1 = b
c2 = copy.copy(b)
c3 = copy.deepcopy(b)
a[-1] = ['BG']
b.append('ccc')

五、枚举值的特点:

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中已经将堆区间的内存的开辟和释放自动化了
当给变量赋值的时候,系统会先在堆区间中自动开辟区间将数据存起来,然后再将数据在堆中的地址存存在
变量中,变量存在栈区间
注意:数字和字符串数据在开辟空间的时候,会先检查内存中之前是否已经有这个数据,如果有就直接使用之前的
数据,没有才开辟新的空间保存数据

七、内存的释放

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

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

推荐阅读更多精彩内容