內(nèi)存管理和拷貝_筆記

一、review

1.寫(xiě)一個(gè)撲克類, 要求擁有發(fā)牌和洗牌的功能(具體的屬性和其他功能自己根據(jù)實(shí)際情況發(fā)揮)

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)

    # 斗地主發(fā)牌方式
    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、(嘗試)寫(xiě)一個(gè)類,其功能是:1.解析指定的歌詞文件的內(nèi)容 2.按時(shí)間顯示歌詞

提示:歌詞文件的內(nèi)容一般是按下面的格式進(jìn)行存儲(chǔ)的。
歌詞前面對(duì)應(yīng)的是時(shí)間,在對(duì)應(yīng)的時(shí)間點(diǎn)可以顯示對(duì)應(yīng)的歌詞

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('藍(lán)蓮花')
print('===:',ly.get_word(123))
print('===:',ly.get_word(10))
print('===:',ly.get_word(89))

ly2 = LyricAnalysis('一首簡(jiǎn)單的歌')
print('!!!:', ly2.get_word(30))
print('!!!:', ly2.get_word(90))

二、多繼承

Python中的類支持多繼承
多繼承的時(shí)候,子類只能繼承第一個(gè)父類所有的屬性和方法,
后面的父類只能繼承字段和方法,方法名一樣時(shí),只繼承第一個(gè)父類

class Animal(object):
    num = 100
    def __init__(self):
        self.age = 0
        self.gender = '雌'
    @classmethod
    def func1(cls):
        print('動(dòng)物類的類方法')
    def func2(self):
        print('動(dòng)物類中的對(duì)象方法')
class Fly(object):
    name = '飛行器'
    def __init__(self):
        self.height = 100
        self.time = 5
        self.speed = 100
    def func2(self):
        print('飛行的對(duì)象方法')
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)

三、運(yùn)算符重載

1.運(yùn)算符

Python中所有的類型都是類,所有的數(shù)據(jù)都是對(duì)象,
Python中使用任意的運(yùn)算符都是在調(diào)用類中的相應(yīng)方法,每一個(gè)運(yùn)算符對(duì)應(yīng)什么方法是固定的,
某種數(shù)據(jù)是否支持某個(gè)運(yùn)算符操作就看這個(gè)數(shù)據(jù)類型中是否實(shí)現(xiàn)了對(duì)應(yīng)的方法

2、運(yùn)算符重載

——指的是在不同的類中實(shí)現(xiàn)同樣的運(yùn)算符對(duì)應(yīng)的函數(shù)
類的對(duì)象默認(rèn)情況下只支持:==、!=

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 -> 當(dāng)前類的對(duì)象,也是+前面的那個(gè)數(shù)據(jù)
    # other -> +后面的那個(gè)數(shù)據(jù), 類型根據(jù)運(yùn)算規(guī)則的設(shè)計(jì)可以是任何類型的數(shù)據(jù)
    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)
    # 注意: <和>符號(hào)只需要重載其中一個(gè)就可以
    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、直接賦值

將變量中的地址直接賦值給新的變量,賦值后兩個(gè)變量的地址相同

2、淺拷貝

拷貝(不管是淺拷貝還是深拷貝)不同于直接賦值,會(huì)產(chǎn)生新的地址,
字符串、列表、元祖的切片,對(duì)象.copy(),copy模塊中的copy方法都是淺拷貝
淺拷貝只會(huì)拷貝當(dāng)前對(duì)象,不會(huì)對(duì)子對(duì)象進(jìn)行拷貝

3、深拷貝

copy模塊中的deepcopy方法是深拷貝,不僅會(huì)拷貝當(dāng)前對(duì)象,還會(huì)拷貝子對(duì)象,產(chǎn)生新的地址

# 練習(xí):
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')

五、枚舉值的特點(diǎn):

1)可以通過(guò)有意義的屬性名直接顯示數(shù)據(jù)

2)每個(gè)數(shù)據(jù)的值不能修改

3)可以做到不同的數(shù)據(jù)的值唯一

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)

六、內(nèi)存的開(kāi)辟

內(nèi)存區(qū)間分為棧區(qū)間和堆區(qū)間,棧區(qū)間的內(nèi)存自動(dòng)開(kāi)辟和釋放,堆區(qū)間的內(nèi)存需要手動(dòng)開(kāi)辟和釋放,
但是Python中已經(jīng)將堆區(qū)間的內(nèi)存的開(kāi)辟和釋放自動(dòng)化了
當(dāng)給變量賦值的時(shí)候,系統(tǒng)會(huì)先在堆區(qū)間中自動(dòng)開(kāi)辟區(qū)間將數(shù)據(jù)存起來(lái),然后再將數(shù)據(jù)在堆中的地址存存在
變量中,變量存在棧區(qū)間
注意:數(shù)字和字符串?dāng)?shù)據(jù)在開(kāi)辟空間的時(shí)候,會(huì)先檢查內(nèi)存中之前是否已經(jīng)有這個(gè)數(shù)據(jù),如果有就直接使用之前的
數(shù)據(jù),沒(méi)有才開(kāi)辟新的空間保存數(shù)據(jù)

七、內(nèi)存的釋放

棧區(qū)間:全局棧區(qū)間在程序結(jié)束后銷毀,函數(shù)棧區(qū)間在函數(shù)調(diào)用結(jié)束后銷毀
堆區(qū)間:看一個(gè)對(duì)象是否銷毀,就看這個(gè)對(duì)象的引用計(jì)數(shù)是否為0,如果引用為0,這個(gè)對(duì)象就會(huì)銷毀(垃圾回收)
注意:Python中針對(duì)對(duì)象的循環(huán)引用已經(jīng)做了處理,程序員不需要寫(xiě)額外的代碼來(lái)解決循環(huán)引用問(wèn)題

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Swift1> Swift和OC的區(qū)別1.1> Swift沒(méi)有地址/指針的概念1.2> 泛型1.3> 類型嚴(yán)謹(jǐn) 對(duì)...
    cosWriter閱讀 11,663評(píng)論 1 32
  • 內(nèi)存管理和拷貝 review 1.內(nèi)置類屬性: name、doc、class、dict、module、bases2...
    漫磋嗟閱讀 202評(píng)論 0 0
  • 前言 把《C++ Primer》[https://book.douban.com/subject/25708312...
    尤汐Yogy閱讀 9,679評(píng)論 1 51
  • 01 - 多繼承 python中的類支持多繼承(讓一個(gè)類同時(shí)繼承多個(gè)類);多繼承的時(shí)候,子類只能繼承第一個(gè)父類所有...
    Tianxiaoyu閱讀 176評(píng)論 0 0
  • 你準(zhǔn)備好了嗎 我的姐妹 我自己 當(dāng)你決定做一件事情的那一刻 你必須全力以赴 沒(méi)有退路 因?yàn)闄C(jī)會(huì)的可貴 也因?yàn)樗闹?..
    摘不下殼子的蝸牛閱讀 328評(píng)論 0 1

友情鏈接更多精彩內(nèi)容