python復(fù)合條件下的字典排序
知乎上有人說,Python3.6以后字典有序且更高效了。群里有同學(xué)推薦了這篇文章給我看,并咨詢字典排序的問題。
大致瀏覽了一下,我當(dāng)即表示不能認(rèn)同這個(gè)說法。這篇文章的作者,應(yīng)該是一位資深的專業(yè)人士,對于Python解釋器如何實(shí)現(xiàn)字典存儲(chǔ)和檢索有著深刻地理解。但他犯了一明顯的常識(shí)性錯(cuò)誤:在邏輯上,字典是數(shù)據(jù)的無序集合,僅依賴于鍵檢索。我們說字典是無序,不是指字典在物理實(shí)體上實(shí)現(xiàn)的時(shí)候真的無序,而是指它的順序?qū)τ脩舳詻]有明確的界定,不能作為數(shù)據(jù)的特性使用。知乎上這篇文章講的字典有序,是指字典在物理實(shí)體上實(shí)現(xiàn)時(shí)的有序,而非邏輯上的有序。
既然字典是無序的,為什么還有那么多討論字典排序的話題呢?其實(shí),在Py2時(shí)代,就存在有序字典(orderdict),但有序字典和我們討論的字典,并非一碼事兒。所謂的字典排序,實(shí)質(zhì)上是根據(jù)排序規(guī)則將字典的鍵排序,得到的排序結(jié)果是一個(gè)列表。
我們用一個(gè)例子來演示一下字典排序:roster是一個(gè)保存學(xué)生信息的字典,請按照女生優(yōu)先、低年級(jí)在前、總成績從高到底排序;如果總成績相同,則順序比較語文、數(shù)學(xué)、英語成績,高者在前。
roster = {
'李妍可': {'性別':'女', '年級(jí)':3, '語文':98, '數(shù)學(xué)':95, '英語':100},
'鄔勝杰': {'性別':'男', '年級(jí)':5, '語文':95, '數(shù)學(xué)':100, '英語':97},
'白星瑤': {'性別':'女', '年級(jí)':2, '語文':100, '數(shù)學(xué)':99, '英語':100},
'吳詩涵': {'性別':'男', '年級(jí)':3, '語文':98, '數(shù)學(xué)':92, '英語':90},
'莊嘉順': {'性別':'男', '年級(jí)':5, '語文':97, '數(shù)學(xué)':95, '英語':100}
}
Python最常用的排序函數(shù)是sorted(),我們就用sorted()來實(shí)現(xiàn)這個(gè)排序。如果一次寫出復(fù)合排序條件,有一定難度。我們化繁為簡,一步步實(shí)現(xiàn)。
1. 比較總成績
>>> sorted(roster, key=lambda name:roster[name]['語文']+roster[name]['數(shù)學(xué)']+roster[name]['英語']) ['吳詩涵', '鄔勝杰', '莊嘉順', '李妍可', '白星瑤']
看起來沒有問題,但sorted默認(rèn)是升序,總成績從高到底排序的話,要使用reverse=True這個(gè)參數(shù)。
>>> sorted(roster, key=lambda name:roster[name]['語文']+roster[name]['數(shù)學(xué)']+roster[name]['英語'], reverse=True) ['白星瑤', '李妍可', '鄔勝杰', '莊嘉順', '吳詩涵']
2. 再來嘗試女生優(yōu)先、低年級(jí)在前的兩個(gè)條件排序
只要在lambda函數(shù)中,把排序項(xiàng)并列寫出來,sorted()就會(huì)自動(dòng)實(shí)現(xiàn)符合條件排序。這里性別排序的條件是'性別'==‘男',對女生而言,結(jié)果是False(0),小于男生的True(1),自然就排在了前面。
>>> sorted(roster, key=lambda name:(roster[name]['性別']=='男',roster[name]['年級(jí)'])) ['白星瑤', '李妍可', '吳詩涵', '鄔勝杰', '莊嘉順']
3. 最終實(shí)現(xiàn)
嘗試了單個(gè)條件和兩個(gè)條件的排序之后,實(shí)現(xiàn)本題目的最終要求就很容易了。不過,成績降序排列的話,不能直接使用reverse=True,因?yàn)闀?huì)影響性別和年級(jí)的排序。我們可以稍微變通一下,達(dá)到最終的目的。
>>> sorted(roster, key=lambda name:(
roster[name]['性別']=='男',
roster[name]['年級(jí)'],
300-roster[name]['語文']-roster[name]['數(shù)學(xué)']-roster[name]['英語'],
100-roster[name]['語文'],
100-roster[name]['數(shù)學(xué)'],
100-roster[name]['英語']
))
['白星瑤', '李妍可', '吳詩涵', '莊嘉順', '鄔勝杰']
到此這篇關(guān)于python復(fù)合條件下的字典排序的文章就介紹到這了,更多相關(guān)python 字典排序內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
django實(shí)現(xiàn)悲觀鎖樂觀鎖的項(xiàng)目實(shí)踐
在Django中,我們可以通過實(shí)現(xiàn)悲觀鎖和樂觀鎖來保證數(shù)據(jù)的安全性,本文就來介紹一下django實(shí)現(xiàn)悲觀鎖樂觀鎖的項(xiàng)目實(shí)踐,感興趣的可以了解一下2023-08-08
Python3隨機(jī)漫步生成數(shù)據(jù)并繪制
這篇文章主要為大家詳細(xì)介紹了Python3隨機(jī)漫步生成數(shù)據(jù)并繪制的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-08-08
python通過自定義isnumber函數(shù)判斷字符串是否為數(shù)字的方法
這篇文章主要介紹了python通過自定義isnumber函數(shù)判斷字符串是否為數(shù)字的方法,涉及Python操作字符串判斷的相關(guān)技巧,需要的朋友可以參考下2015-04-04
Python實(shí)現(xiàn)的簡單計(jì)算器功能詳解
這篇文章主要介紹了Python實(shí)現(xiàn)的簡單計(jì)算器功能,結(jié)合實(shí)例形式詳細(xì)分析了Python實(shí)現(xiàn)計(jì)算器功能的具體步驟、相關(guān)操作技巧與注意事項(xiàng),需要的朋友可以參考下2018-08-08
Python OpenCV高斯金字塔與拉普拉斯金字塔的實(shí)現(xiàn)
這篇文章主要介紹了Python OpenCV高斯金字塔與拉普拉斯金字塔的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03
Python實(shí)現(xiàn)字符串匹配算法代碼示例
這篇文章主要介紹了Python實(shí)現(xiàn)字符串匹配算法代碼示例,涉及字符串匹配存在的問題,蠻力法字符串匹配,Horspool算法,具有一定參考價(jià)值,需要的朋友可以了解下。2017-12-12
Python繪制數(shù)據(jù)動(dòng)態(tài)圖的方法詳解
這篇文章主要為大家詳細(xì)介紹了如何利用Python語言繪制好看的數(shù)據(jù)動(dòng)態(tài)圖,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起動(dòng)手嘗試一下2022-07-07

