什么是迭代
在Python中,如果給定一個(gè)list或tuple剑肯,我們可以通過for循環(huán)來遍歷這個(gè)list或tuple沛婴,這種遍歷我們成為迭代(Iteration)。
在Python中昔汉,迭代是通過for ... in來完成的懈万,而很多語言比如C或者Java,迭代list是通過下標(biāo)完成的靶病,比如Java代碼:
for (i=0; i
n = list[i]
}
可以看出会通,Python的for循環(huán)抽象程度要高于Java的for循環(huán)。
因?yàn)?Python 的 for循環(huán)不僅可以用在list或tuple上娄周,還可以作用在其他任何可迭代對象上涕侈。
因此,迭代操作就是對于一個(gè)集合煤辨,無論該集合是有序還是無序裳涛,我們用 for 循環(huán)總是可以依次取出集合的每一個(gè)元素。
注意: 集合是指包含一組元素的數(shù)據(jù)結(jié)構(gòu)众辨,我們已經(jīng)介紹的包括:1.有序集合:list端三,tuple,str和unicode泻轰;2.無序集合:set技肩;3.無序集合并且具有 key-value 對:dict
而迭代是一個(gè)動(dòng)詞,它指的是一種操作浮声,在Python中虚婿,就是 for 循環(huán)。
迭代與按下標(biāo)訪問數(shù)組最大的不同是泳挥,后者是一種具體的迭代實(shí)現(xiàn)方式然痊,而前者只關(guān)心迭代結(jié)果,根本不關(guān)心迭代內(nèi)部是如何實(shí)現(xiàn)的屉符。
任務(wù)
請用for循環(huán)迭代數(shù)列 1-100 并打印出7的倍數(shù)剧浸。
索引迭代
Python中锹引,迭代永遠(yuǎn)是取出元素本身,而非元素的索引唆香。
對于有序集合嫌变,元素確實(shí)是有索引的。有的時(shí)候躬它,我們確實(shí)想在 for 循環(huán)中拿到索引腾啥,怎么辦?
方法是使用enumerate() 函數(shù):
>>> L = ['Adam', 'Lisa', 'Bart', 'Paul']
>>>for index, name in enumerate(L):
... ?? ? ? ? print index, '-', name
...
0 - Adam
1 - Lisa
2 - Bart
3 - Paul
使用 enumerate() 函數(shù)冯吓,我們可以在for循環(huán)中同時(shí)綁定索引index和元素name倘待。但是,這不是 enumerate() 的特殊語法组贺。實(shí)際上凸舵,enumerate() 函數(shù)把:
['Adam', 'Lisa', 'Bart', 'Paul']
變成了類似:
[(0, 'Adam'), (1, 'Lisa'), (2, 'Bart'), (3, 'Paul')]
因此,迭代的每一個(gè)元素實(shí)際上是一個(gè)tuple:
for t in enumerate(L):
index = t[0]
name = t[1]
print index, '-', name
如果我們知道每個(gè)tuple元素都包含兩個(gè)元素失尖,for循環(huán)又可以進(jìn)一步簡寫為:
for index, name in enumerate(L):
print index, '-', name
這樣不但代碼更簡單啊奄,而且還少了兩條賦值語句。
可見雹仿,索引迭代也不是真的按索引訪問增热,而是由 enumerate() 函數(shù)自動(dòng)把每個(gè)元素變成 (index, element) 這樣的tuple,再迭代胧辽,就同時(shí)獲得了索引和元素本身峻仇。
任務(wù)
zip()函數(shù)可以把兩個(gè) list 變成一個(gè) list:
>>> zip([10, 20, 30], ['A', 'B', 'C'])
[(10, 'A'), (20, 'B'), (30, 'C')]
在迭代 ['Adam', 'Lisa', 'Bart', 'Paul'] 時(shí),如果我們想打印出名次 - 名字(名次從1開始)邑商,請考慮如何在迭代中打印出來摄咆。
提示:考慮使用zip()函數(shù)和range()函數(shù)
迭代dict的value
我們已經(jīng)了解了dict對象本身就是可迭代對象,用 for 循環(huán)直接迭代 dict人断,可以每次拿到dict的一個(gè)key吭从。
如果我們希望迭代 dict 對象的value,應(yīng)該怎么做恶迈?
dict 對象有一個(gè)values() 方法涩金,這個(gè)方法把dict轉(zhuǎn)換成一個(gè)包含所有value的list,這樣暇仲,我們迭代的就是 dict的每一個(gè) value:
d = { 'Adam': 95, 'Lisa': 85, 'Bart': 59 }print d.values()# [85, 95, 59]forvind.values():print v# 85# 95# 59
如果仔細(xì)閱讀Python的文檔步做,還可以發(fā)現(xiàn),dict除了values()方法外奈附,還有一個(gè)itervalues()方法全度,用itervalues()方法替代values()方法,迭代效果完全一樣:
d = { 'Adam': 95, 'Lisa': 85, 'Bart': 59 }print d.itervalues()# forvind.itervalues():print v# 85# 95# 59
那這兩個(gè)方法有何不同之處呢斥滤?
1.values()方法實(shí)際上把一個(gè) dict 轉(zhuǎn)換成了包含 value 的list将鸵。
2. 但是itervalues()方法不會轉(zhuǎn)換勉盅,它會在迭代過程中依次從 dict 中取出 value,所以 itervalues() 方法比 values() 方法節(jié)省了生成 list 所需的內(nèi)存顶掉。
3.?打印 itervalues() 發(fā)現(xiàn)它返回一個(gè)? 對象草娜,這說明在Python中,for 循環(huán)可作用的迭代對象遠(yuǎn)不止 list痒筒,tuple驱还,str,unicode凸克,dict等,任何可迭代對象都可以作用于for循環(huán)闷沥,而內(nèi)部如何迭代我們通常并不用關(guān)心萎战。
如果一個(gè)對象說自己可迭代,那我們就直接用 for 循環(huán)去迭代它舆逃,可見蚂维,迭代是一種抽象的數(shù)據(jù)操作,它不對迭代對象內(nèi)部的數(shù)據(jù)有任何要求路狮。
任務(wù)
給定一個(gè)dict:
d = { 'Adam': 95, 'Lisa': 85, 'Bart': 59, 'Paul': 74 }
請計(jì)算所有同學(xué)的平均分虫啥。
迭代dict的key和value
我們了解了如何迭代 dict的key和value,那么奄妨,在一個(gè) for 循環(huán)中涂籽,能否同時(shí)迭代 key和value?答案是肯定的砸抛。
首先评雌,我們看看 dict 對象的items()方法返回的值:
>>> d = { 'Adam': 95, 'Lisa': 85, 'Bart': 59 }
>>> print d.items()
[('Lisa', 85), ('Adam', 95), ('Bart', 59)]
可以看到,items() 方法把dict對象轉(zhuǎn)換成了包含tuple的list直焙,我們對這個(gè)list進(jìn)行迭代景东,可以同時(shí)獲得key和value:
>>> for key, value in d.items():
...? ?? print key, ':', value
...
Lisa : 85
Adam : 95
Bart : 59
和 values() 有一個(gè) itervalues() 類似,items()也有一個(gè)對應(yīng)的iteritems()奔誓,iteritems() 不把dict轉(zhuǎn)換成list斤吐,而是在迭代過程中不斷給出 tuple,所以厨喂, iteritems() 不占用額外的內(nèi)存和措。
任務(wù)
請根據(jù)dict:
d = { 'Adam': 95, 'Lisa': 85, 'Bart': 59, 'Paul': 74 }
打印出 name : score,最后再打印出平均分 average : score杯聚。