Python 把兩層列表展開平鋪成一層(5種實(shí)現(xiàn)方式)
這幾天和同事在討論,如何用 Python 寫出優(yōu)雅的讓列表中的列表展開,變成扁平化的列表。
例如
# 期望輸入 input = [[('A', 1), ('B', 2)], [('C', 3), ('D', 4)]] # 期望輸出 output = [('A', 1), ('B', 2), ('C', 3), ('D', 4)]
map 函數(shù)合并
>>> new = []; map(new.extend, input); new [None, None] [('A', 1), ('B', 2), ('C', 3), ('D', 4)]
這個(gè)方法看上去還可以,但是有個(gè)致命的缺點(diǎn),就是map函數(shù)會(huì)返回值,并且這個(gè)返回值是沒有用的。另外還需要提前聲明一個(gè)變量,從代碼的簡(jiǎn)潔性上,不夠簡(jiǎn)潔優(yōu)雅。
sum 函數(shù)合并
>>> sum(input, []) [('A', 1), ('B', 2), ('C', 3), ('D', 4)]
這個(gè)看上去很簡(jiǎn)潔,不過(guò)有類似字符串累加的性能陷阱。后面有性能對(duì)比。
reduce 函數(shù)
>>> reduce(list.__add__, input) [('A', 1), ('B', 2), ('C', 3), ('D', 4)]
做序列的累加操作。也是有累加的性能陷阱。
列表推導(dǎo)式
>>> [item for sublist in input for item in sublist] [('A', 1), ('B', 2), ('C', 3), ('D', 4)]
列表推導(dǎo)式,看著有些長(zhǎng),而且還要for循環(huán)兩次,變成一行理解需要費(fèi)勁一些,沒有那么直觀。
itertools 類庫(kù)
>>> list(itertools.chain(*input)) [('A', 1), ('B', 2), ('C', 3), ('D', 4)]
通過(guò)第三方類庫(kù)類實(shí)現(xiàn)的,相比其他的幾個(gè)實(shí)現(xiàn),看著還算比較優(yōu)雅。最后的性能發(fā)現(xiàn)居然還很高。
性能大對(duì)比
python -mtimeit -s'l=[[1,2,3],[4,5,6], [7], [8,9]]*99' 'reduce(list.__add__,l)' 1000 loops, best of 3: 547 usec per loop python -mtimeit -s'l=[[1,2,3],[4,5,6], [7], [8,9]]*99' 'sum(l, [])' 1000 loops, best of 3: 509 usec per loop python -mtimeit -s'l=[[1,2,3],[4,5,6], [7], [8,9]]*99' '[item for sublist in l for item in sublist]' 10000 loops, best of 3: 52.8 usec per loop python -mtimeit -s'l=[[1,2,3],[4,5,6], [7], [8,9]]*99; import itertools;' 'list(itertools.chain(*l))' 10000 loops, best of 3: 35.9 usec per loop python -mtimeit -s'l=[[1,2,3],[4,5,6], [7], [8,9]]*99' 'new = []; map(new.extend, l); new' 10000 loops, best of 3: 34.1 usec per loop
歡迎大家共同探討優(yōu)雅的的實(shí)現(xiàn)和性能的優(yōu)化。
補(bǔ)充:python 將(含嵌套的)dict平鋪展開
話不多說(shuō),直接上代碼:
def prefix_dict(di_, prefix_s=''): """ 把字典的每個(gè)key都帶上前綴prefix_s :param di_: :param prefix_s: :return: """ return {prefix_s + k: v for k, v in di_.items()} def spear_dict(di_, con_s='.'): """ 展開dict(如果下層還是dict),需要遞歸,展開到下層的數(shù)據(jù)類型不是字典為止 可能實(shí)用的地方:將文檔類的數(shù)據(jù)格式化成更加關(guān)系化的樣子可能有用 :param di_: 輸入字典 :param con_s: 層級(jí)間的連接符號(hào) :return: 深度不大于1的字典,嵌套的其他數(shù)據(jù)類型照舊 """ ret_di = {} for k, v in di_.items(): if type(v) is dict: v = spear_dict(v) # 這里或許有不寫到這一層的更好寫法 # for k_, v_ in v.items(): # ret_di.update({con_s.join([k, k_]): v_}) ret_di.update(prefix_dict(v, prefix_s=k + con_s)) else: ret_di.update({k: v}) return ret_di
>>> di_ {'title': '新田商業(yè)街', 'reliability': 7, 'addressComponents': {'streetNumber': '', 'city': '深圳市', 'street': '', 'province': '廣東省', 'district': '龍華區(qū)'}, 'location': {'lng': 114.09127044677734, 'lat': 22.700519561767578}, 'adInfo': {'adcode': '440309'}, 'level': 11, 'more_deep': {'loca': {'lng': 114.09127044677734, 'lat': 22.700519561767578}}} >>> spear_dict(di_) {'title': '新田商業(yè)街', 'reliability': 7, 'addressComponents.streetNumber': '', 'addressComponents.city': '深圳市', 'addressComponents.street': '', 'addressComponents.province': '廣東省', 'addressComponents.district': '龍華區(qū)', 'location.lng': 114.09127044677734, 'location.lat': 22.700519561767578, 'adInfo.adcode': '440309', 'level': 11, 'more_deep.loca.lng': 114.09127044677734, 'more_deep.loca.lat': 22.700519561767578} spear_dict(di_, '_') {'title': '新田商業(yè)街', 'reliability': 7, 'addressComponents_streetNumber': '', 'addressComponents_city': '深圳市', 'addressComponents_street': '', 'addressComponents_province': '廣東省', 'addressComponents_district': '龍華區(qū)', 'location_lng': 114.09127044677734, 'location_lat': 22.700519561767578, 'adInfo_adcode': '440309', 'level': 11, 'more_deep_loca.lng': 114.09127044677734, 'more_deep_loca.lat': 22.700519561767578}
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
- python中必會(huì)的四大高級(jí)數(shù)據(jù)類型(字符,元組,列表,字典)
- Python列表刪除重復(fù)元素與圖像相似度判斷及刪除實(shí)例代碼
- Python 統(tǒng)計(jì)列表中重復(fù)元素的個(gè)數(shù)并返回其索引值的實(shí)現(xiàn)方法
- Python基礎(chǔ)詳解之列表復(fù)制
- 淺談Python列表嵌套字典轉(zhuǎn)化的問(wèn)題
- Python隨機(jī)函數(shù)random隨機(jī)獲取數(shù)字、字符串、列表等使用詳解
- Python列表排序方法reverse、sort、sorted詳解
- Python3 列表list合并的4種方法
- python獲取指定時(shí)間段內(nèi)特定規(guī)律的日期列表
- python實(shí)現(xiàn)合并兩個(gè)有序列表的示例代碼
- python求列表對(duì)應(yīng)元素的乘積和的實(shí)現(xiàn)
- Python統(tǒng)計(jì)列表元素出現(xiàn)次數(shù)的方法示例
- python 合并列表的八種方法
- python 列表元素左右循環(huán)移動(dòng) 的多種解決方案
- Python列表排序 list.sort方法和內(nèi)置函數(shù)sorted用法
- 淺談Python基礎(chǔ)之列表那些事兒
相關(guān)文章
Python讀取xlsx文件的實(shí)現(xiàn)方法
這篇文章主要介紹了Python讀取xlsx文件的實(shí)現(xiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07python中執(zhí)行shell命令的幾個(gè)方法小結(jié)
這篇文章主要介紹了python中執(zhí)行shell命令的幾個(gè)方法,本文一共給出3種方法實(shí)現(xiàn)執(zhí)行shell命令,需要的朋友可以參考下2014-09-09在python中利用KNN實(shí)現(xiàn)對(duì)iris進(jìn)行分類的方法
今天小編就為大家分享一篇在python中利用KNN實(shí)現(xiàn)對(duì)iris進(jìn)行分類的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-12-12用Python畫一個(gè)LinkinPark的logo代碼實(shí)例
這篇文章主要介紹了用Python畫一個(gè)LinkinPark的logo代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-09-09Python中scatter函數(shù)參數(shù)及用法詳解
這篇文章主要介紹了Python中scatter函數(shù)參數(shù)及用法詳解,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-11-11python實(shí)現(xiàn)跨進(jìn)程(跨py文件)通信示例
本文主要介紹了python實(shí)現(xiàn)跨進(jìn)程(跨py文件)通信示例,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03