Python 中的容器 collections

Python 中的容器 collections

寫(xiě)在之前

我們都知道 Python 中內(nèi)置了許多標(biāo)準(zhǔn)的數(shù)據(jù)結(jié)構(gòu)胀瞪,比如列表熏瞄,元組脚祟,字典等。與此同時(shí)標(biāo)準(zhǔn)庫(kù)還提供了一些額外的數(shù)據(jù)結(jié)構(gòu)强饮,我們可以基于它們創(chuàng)建所需的新數(shù)據(jù)結(jié)構(gòu)由桌。

Python 附帶了一個(gè)「容器」模塊 collections,它包含了很多的容器數(shù)據(jù)類(lèi)型邮丰,今天我們來(lái)討論其中幾個(gè)常用的容器數(shù)據(jù)類(lèi)型行您,掌握了這幾個(gè)可以減少我們重復(fù)造輪子所帶來(lái)的煩擾。

namedtuple

相信你已經(jīng)熟悉了元組剪廉,一個(gè)元組相當(dāng)于一個(gè)不可變的列表娃循,你可以存儲(chǔ)一個(gè)數(shù)據(jù)的序列。這里要說(shuō)的 namedtuple(命名元組)和元組非常像斗蒋,它們都不能修改自己的數(shù)據(jù)捌斧。說(shuō)完了像,那么它們有哪些地方不像呢泉沾?

作為元組捞蚂,為了獲取其中的數(shù)據(jù),我們需要使用整數(shù)作為索引:

>>> people = ('Rocky', 'python')
>>> print(people[0])
Rocky

而 namedtuple 把元組變成了一個(gè)針對(duì)簡(jiǎn)單任務(wù)的容器跷究,我們不必使用整數(shù)索引來(lái)訪問(wèn) namedtuple 的數(shù)據(jù)姓迅,反而可以像用字典一樣訪問(wèn) namedtuple。

>>> from collections import namedtuple
>>> people = namedtuple('people', 'name age like')
>>> Rocky = people(name = 'rocky', age = 23, like = 'python')
>>> print(Rocky)
people(name='rocky', age=23, like='python')
>>> print(Rocky.name)
rocky

一個(gè) namedtuple 有兩個(gè)必須的參數(shù):元組名稱(chēng)和字段名稱(chēng)揭朝。在上面的代碼中队贱,我們的元組名稱(chēng)是 people,字段名稱(chēng)是 name潭袱,age柱嫌,like。nametuple 讓元組變的更加易讀屯换,很容易理解代碼是做什么的编丘,同樣我們也不用使用整數(shù)索引來(lái)訪問(wèn)一個(gè)命名元組(上面代碼我們用 name 訪問(wèn)了 namedtuple 中的數(shù)據(jù))与学,這讓我們的代碼更加容易維護(hù)。

但是你一定要記住的是嘉抓,雖然它的用法很爽索守,但它還是一個(gè)元組!所以屬性值在 namedtuple 中是不可變的抑片。

我們?cè)谏厦嬲f(shuō)過(guò)可以像用字典一樣訪問(wèn) namedtuple卵佛,那么當(dāng)然也可以把它轉(zhuǎn)為字典,具體操作如下所示:

>>> from collections import namedtuple
>>> people = namedtuple('people', 'name age like')
>>> Rocky = people(name = 'rocky', age = 23, like = 'python')
>>> print(Rocky._asdict())
OrderedDict([('name', 'rocky'), ('age', 23), ('like', 'python')])

defaultdict

我之前在使用字典的時(shí)候相當(dāng)隨意敞斋,只是隨便 dict 一下就好了截汪,然而這樣使用存在一個(gè)問(wèn)題:當(dāng)使用的 key 不存在的時(shí)候會(huì)報(bào) KeyError,而 defaultdict 就比較厲害了植捎,我們完全不需要檢查 key 是否存在衙解,所以我們能像下面這樣做的隨心所欲:

from collections import defaultdict
languages = (
 ('rocky', 'python'),
 ('snow', 'c'),
 ('leey', 'java'),
 ('rocky', 'c++'),
 ('leey', 'c#')
)
favourite = defaultdict(list)
for name, language in languages:
 favourite[name].append(language)
print(favourite)

輸出如下所示:

defaultdict(<type 'list'>, {'leey': ['java', 'c#'], 'rocky': ['python', 'c++'], 'snow': ['c']})

然后我們?cè)倩氐健版I不存在,會(huì)觸發(fā) KeyError 異逞媸啵”這個(gè)問(wèn)題上來(lái)蚓峦,我們先來(lái)看 dict 觸發(fā) KeyError 的例子:

my_dict = {}
my_dict['name']['like'] = 'python'

輸出如下:

KeyError: 'name'

defaultdict 則用了一個(gè)非常巧妙的方式繞過(guò)了這個(gè)問(wèn)題,請(qǐng)看下面的操作:

import collections
language = lambda : collections.defaultdict(language)
my_dict = language()
my_dict['name']['like'] = 'python'

運(yùn)行一下顯示正常济锄,我們可以用 json.dumps 打印出 my_dict 的內(nèi)容:

import json
print(json.dumps(my_dict))

運(yùn)行結(jié)果如下:

{"name": {"like": "python"}}

Counter

Counter 是一個(gè)計(jì)數(shù)器暑椰,它可以幫助我們針對(duì)某項(xiàng)數(shù)據(jù)進(jìn)行計(jì)數(shù),比如可以用它來(lái)統(tǒng)計(jì)每個(gè)人擅長(zhǎng)的編程語(yǔ)言:

from collections import Counter
languages = (
 ('rocky', 'python'),
 ('snow', 'c'),
 ('leey', 'java'),
 ('rocky', 'c++'),
 ('leey', 'c#')
)
cnt = Counter(name for name, language in languages)
print(cnt)

運(yùn)行結(jié)果如下所示:

Counter({'leey': 2, 'rocky': 2, 'snow': 1})

當(dāng)然我們也可以用它來(lái)統(tǒng)計(jì)一個(gè)文件荐绝,比如:

from collections import Counter
with open('test.txt', 'rb') as f:
 line_cnt = Counter(f)
print(line_cnt)

deque

deque 提供了一個(gè)雙端隊(duì)列干茉,我們可以在首尾兩端添加或者刪除元素,在前面的文章中(Python 標(biāo)準(zhǔn)庫(kù)之雙端隊(duì)列)寫(xiě)過(guò)很泊,沒(méi)看過(guò)的可以看看角虫。

想要使用 deque,首先我們要從 collections 中導(dǎo)入 deque 模塊委造,然后創(chuàng)建一個(gè) deque 對(duì)象戳鹅,它的用法就像我們前面學(xué)過(guò)的 list 一樣,并且提供了類(lèi)似的方法昏兆,具體如下所示:

from collections import deque
deq = deque()
deq.append(1)
deq.append(2)
deq.append(3)
print(deq)
print(len(deq))
print(deq[0])
print(deq[-1])

輸出結(jié)果如下:

deque([1, 2, 3])
3
1
3

我們可以從兩端取出數(shù)據(jù):

from collections import deque
deq = deque(range(5))
print('len(deq) == {}'.format(len(deq)))
deq.popleft()
deq.pop()
print(deq)

輸出的結(jié)果如下所示:

len(deq) == 5
deq == deque([1, 2, 3])

我們也可以對(duì)這個(gè)列表的大小進(jìn)行限制枫虏,當(dāng)超出我們的限制的時(shí)候,數(shù)據(jù)會(huì)從另一端被 pop 出去爬虱,具體我們來(lái)看下面的操作:

from collections import deque
deq = deque(maxlen=3)
deq.append(1)
deq.append(2)
deq.append(3)
print(deq)
deq.append(4)
print (deq)

輸出的結(jié)果如下:

deque([1, 2, 3], maxlen=3)
deque([2, 3, 4], maxlen=3)

當(dāng)超出 maxlen 的值時(shí)隶债,最左邊的數(shù)據(jù)將從隊(duì)列中刪除。

當(dāng)然我們還可以從任意一端擴(kuò)展這個(gè)雙端隊(duì)列中的數(shù)據(jù):

from collections import deque
deq = deque([1,2,3])
deq.extendleft([0])
deq.extend([4,5,6])
print(deq)

輸出的結(jié)果如下所示:

deque([0, 1, 2, 3, 4, 5, 6])

如果有那些不足的地方跑筝,歡迎大家補(bǔ)充死讹!

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市曲梗,隨后出現(xiàn)的幾起案子赞警,更是在濱河造成了極大的恐慌妓忍,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,695評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件愧旦,死亡現(xiàn)場(chǎng)離奇詭異世剖,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)笤虫,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,569評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén)旁瘫,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人琼蚯,你說(shuō)我怎么就攤上這事境蜕。” “怎么了凌停?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,130評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)售滤。 經(jīng)常有香客問(wèn)我罚拟,道長(zhǎng),這世上最難降的妖魔是什么完箩? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,648評(píng)論 1 297
  • 正文 為了忘掉前任赐俗,我火速辦了婚禮,結(jié)果婚禮上弊知,老公的妹妹穿的比我還像新娘阻逮。我一直安慰自己,他們只是感情好秩彤,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,655評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布叔扼。 她就那樣靜靜地躺著,像睡著了一般漫雷。 火紅的嫁衣襯著肌膚如雪瓜富。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 52,268評(píng)論 1 309
  • 那天降盹,我揣著相機(jī)與錄音与柑,去河邊找鬼。 笑死蓄坏,一個(gè)胖子當(dāng)著我的面吹牛价捧,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播涡戳,決...
    沈念sama閱讀 40,835評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼结蟋,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了渔彰?” 一聲冷哼從身側(cè)響起椎眯,我...
    開(kāi)封第一講書(shū)人閱讀 39,740評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤挠将,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后编整,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體舔稀,經(jīng)...
    沈念sama閱讀 46,286評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,375評(píng)論 3 340
  • 正文 我和宋清朗相戀三年掌测,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了内贮。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,505評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡汞斧,死狀恐怖夜郁,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情粘勒,我是刑警寧澤竞端,帶...
    沈念sama閱讀 36,185評(píng)論 5 350
  • 正文 年R本政府宣布,位于F島的核電站庙睡,受9級(jí)特大地震影響事富,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜乘陪,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,873評(píng)論 3 333
  • 文/蒙蒙 一统台、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧啡邑,春花似錦贱勃、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,357評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至流部,卻和暖如春拔鹰,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背贵涵。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,466評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工列肢, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人宾茂。 一個(gè)月前我還...
    沈念sama閱讀 48,921評(píng)論 3 376
  • 正文 我出身青樓瓷马,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親跨晴。 傳聞我的和親對(duì)象是個(gè)殘疾皇子欧聘,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,515評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容