點(diǎn)評(píng):Python標(biāo)準(zhǔn)庫(kù)的collections模塊提供了很多有用的數(shù)據(jù)結(jié)構(gòu),這些內(nèi)容并不是每個(gè)開(kāi)發(fā)者都清楚,就比如題目問(wèn)到的namedtuple,在我參加過(guò)的面試中,90%的面試者都不能準(zhǔn)確的說(shuō)出它的作用和應(yīng)用場(chǎng)景。
此外,deque也是一個(gè)非常有用但又經(jīng)常被忽視的類(lèi),還有Counter、OrderedDict 、defaultdict 、UserDict等類(lèi),大家清楚它們的用法嗎?
在使用面向?qū)ο缶幊陶Z(yǔ)言的時(shí)候,定義類(lèi)是最常見(jiàn)的一件事情,有的時(shí)候,我們會(huì)用到只有屬性沒(méi)有方法的類(lèi),這種類(lèi)的對(duì)象通常只用于組織數(shù)據(jù),并不能接收消息,所以我們把這種類(lèi)稱(chēng)為數(shù)據(jù)類(lèi)或者退化的類(lèi),就像C語(yǔ)言中的結(jié)構(gòu)體那樣。
我們并不建議使用這種退化的類(lèi),在Python中可以用 namedtuple(命名元組)來(lái)替代這種類(lèi)。
from collections import namedtuple Card = namedtuple('Card', ('suite', 'face')) card1 = Card('紅桃', 13) card2 = Card('草花', 5) print(f'{card1.suite}{card1.face}') print(f'{card2.suite}{card2.face}') 命名元組與普通元組一樣是不可變?nèi)萜鳎坏?shù)據(jù)存儲(chǔ)在namedtuple的頂層屬性中,數(shù)據(jù)就不能再修改了,也就意味著對(duì)象上的所有屬性都遵循“一次寫(xiě)入,多次讀取”的原則。
和普通元組不同的是,命名元組中的數(shù)據(jù)有訪問(wèn)名稱(chēng),可以通過(guò)名稱(chēng)而不是索引來(lái)獲取保存的數(shù)據(jù),不僅在操作上更加簡(jiǎn)單,代碼的可讀性也會(huì)更好。命名元組的本質(zhì)就是一個(gè)類(lèi),所以它還可以作為父類(lèi)創(chuàng)建子類(lèi)。除此之外,命名元組內(nèi)置了一系列的方法,例如,可以通過(guò)_asdict方法將命名元組處理成字典,也可以通過(guò)_replace方法創(chuàng)建命名元組對(duì)象的淺拷貝。
class MyCard(Card): def show(self): faces = ['', 'A', '2', '3', '4', '5', '6', '7', '8', '9', '10', 'J', 'Q', 'K'] return f'{self.suite}{faces[self.face]}' print(Card) # card3 = MyCard('方塊', 12) print(card3.show())
# 方塊Q print(dict(card1._asdict()))
# {'suite': '紅桃', 'face': 13} print(card2._replace(suite='方塊'))
# Card(suite='方塊', face=5)
總而言之,命名元組能更好的組織數(shù)據(jù)結(jié)構(gòu),讓代碼更加清晰和可讀,在很多場(chǎng)景下是元組、字典和數(shù)據(jù)類(lèi)的替代品。在需要?jiǎng)?chuàng)建占用空間更少的不可變類(lèi)時(shí),命名元組就是很好的選擇。