python自定義排序

cmp_to_key

入門級(jí)排序

python的列表提供了sort方法己儒,下面是該方法的一個(gè)示例

lst = [(9, 4), (2, 10), (4, 3), (3, 6)]
lst.sort(key=lambda item: item[0])
print(lst)

sort方法的key參數(shù)需要設(shè)置一個(gè)函數(shù),這個(gè)函數(shù)返回元素參與大小比較的值巾腕,這看起來(lái)沒(méi)有問(wèn)題面睛,但如果想實(shí)現(xiàn)更加復(fù)雜的自定義排序,就不那么容易了尊搬。

高階排序

前面例子的排序規(guī)則是根據(jù)元組里第一個(gè)元素的大小進(jìn)行排序叁鉴,我現(xiàn)在修改規(guī)則,如果元組里第一個(gè)元素是奇數(shù)毁嗦,就用元組里第一個(gè)元素進(jìn)行排序亲茅,如果元組里第一個(gè)元素是偶數(shù),則用這個(gè)元組里的第二個(gè)元素進(jìn)行大小比較狗准,面對(duì)這樣的需求克锣,列表的sort方法無(wú)法滿足。

對(duì)于這種情形腔长,可以使用 functools.cmp_to_key 來(lái)解決

from functools import cmp_to_key
lst = [(9, 4), (2, 10), (4, 3), (3, 6)]

def cmp(x, y):
    a = x[0] if x[0] %2 == 1 else x[1]
    b = y[0] if y[0] %2 == 1 else y[1]

    return 1 if a > b else -1 if a < b else 0

lst.sort(key=cmp_to_key(cmp))
print(lst)

仍然使用sort進(jìn)行排序袭祟,我實(shí)現(xiàn)了一個(gè)cmp函數(shù),該函數(shù)實(shí)現(xiàn)了需求中所提到的要求捞附,該函數(shù)最終要返回兩個(gè)元組比較的大小關(guān)系

示例

from functools import cmp_to_key 
L=[9,2,23,1,2]
 
sorted(L,key=cmp_to_key(lambda x,y:y-x))
輸出:
[23, 9, 2, 2, 1]
 
 
sorted(L,key=cmp_to_key(lambda x,y:x-y))
輸出:
[1, 2, 2, 9, 23]

cmp_to_key的實(shí)現(xiàn)

其實(shí)cmp_to_key的實(shí)現(xiàn)非常簡(jiǎn)單

def cmp_to_key(mycmp):
    """Convert a cmp= function into a key= function"""
    class K(object):
        __slots__ = ['obj']
        def __init__(self, obj):
            self.obj = obj
        def __lt__(self, other):
            return mycmp(self.obj, other.obj) < 0
        def __gt__(self, other):
            return mycmp(self.obj, other.obj) > 0
        def __eq__(self, other):
            return mycmp(self.obj, other.obj) == 0
        def __le__(self, other):
            return mycmp(self.obj, other.obj) <= 0
        def __ge__(self, other):
            return mycmp(self.obj, other.obj) >= 0
        __hash__ = None
    return K

它在內(nèi)部定義了一個(gè)類K巾乳, 并使用我傳入的cmp函數(shù)完成了比較關(guān)系運(yùn)算符的重載,函數(shù)返回的是一個(gè)類鸟召,而sort函數(shù)的key需要的是一個(gè)函數(shù)胆绊,看起來(lái)矛盾,但在python中欧募,這樣做完全可行压状,因?yàn)轭惡秃瘮?shù)都是callable的,這里把類當(dāng)成了函數(shù)來(lái)用跟继。

在本篇第一段代碼中

lst.sort(key=lambda item: item[0])

lambda表達(dá)式生成的匿名函數(shù)返回的是元組的第一個(gè)元素進(jìn)行大小比較种冬,而現(xiàn)在,cmp_to_key返回的是類K舔糖,參與比較的是K的對(duì)象娱两,由于K已經(jīng)實(shí)現(xiàn)了比較關(guān)系運(yùn)算符重載,且算法就是我剛剛實(shí)現(xiàn)的cmp函數(shù)金吗,這樣就最終實(shí)現(xiàn)了自定義排序十兢。

實(shí)例

力扣題目:179.最大數(shù)

給定一組非負(fù)整數(shù) nums趣竣,重新排列每個(gè)數(shù)的順序(每個(gè)數(shù)不可拆分)使之組成一個(gè)最大的整數(shù)。
注意:輸出結(jié)果可能非常大纪挎,所以你需要返回一個(gè)字符串而不是整數(shù)期贫。

示例 1:
輸入:nums = [10,2]
輸出:"210"

示例 2:
輸入:nums = [3,30,34,5,9]
輸出:"9534330"

示例 3:
輸入:nums = [1]
輸出:"1"

示例 4:
輸入:nums = [10]
輸出:"10"

提示:

1 <= nums.length <= 100
0 <= nums[i] <= 109

來(lái)源:力扣(LeetCode)
鏈接:https://leetcode-cn.com/problems/largest-number
著作權(quán)歸領(lǐng)扣網(wǎng)絡(luò)所有。商業(yè)轉(zhuǎn)載請(qǐng)聯(lián)系官方授權(quán)异袄,非商業(yè)轉(zhuǎn)載請(qǐng)注明出>處通砍。

題目分析

這道題中列表nums中兩個(gè)值的相對(duì)位置并不能由單一num決定,而是說(shuō) x與y拼接比y與x拼接的值大烤蜕,那么就用[x,y]的順序封孙,否則用[y,x]的順序。此時(shí)就是所謂的:?jiǎn)蝹€(gè)元素并沒(méi)有一個(gè)絕對(duì)的大小的情況

代碼

from functools import cmp_to_key

class Solution:
    def largestNumber(self, nums):
        ret = map(str, nums)

        def cmp(a, b):
            if a + b >= b + a:
                return 1
            else:
                return -1
        ret = sorted(ret, key=cmp_to_key(cmp), reverse=True)
        return ''.join(ret) if ret[0] != '0' else '0'
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末讽营,一起剝皮案震驚了整個(gè)濱河市虎忌,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌橱鹏,老刑警劉巖膜蠢,帶你破解...
    沈念sama閱讀 206,968評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異莉兰,居然都是意外死亡挑围,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門糖荒,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)杉辙,“玉大人,你說(shuō)我怎么就攤上這事捶朵≈┦福” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 153,220評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵综看,是天一觀的道長(zhǎng)品腹。 經(jīng)常有香客問(wèn)我,道長(zhǎng)红碑,這世上最難降的妖魔是什么珍昨? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,416評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮句喷,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘兔毙。我一直安慰自己唾琼,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布澎剥。 她就那樣靜靜地躺著锡溯,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上祭饭,一...
    開(kāi)封第一講書(shū)人閱讀 49,144評(píng)論 1 285
  • 那天芜茵,我揣著相機(jī)與錄音,去河邊找鬼倡蝙。 笑死九串,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的寺鸥。 我是一名探鬼主播猪钮,決...
    沈念sama閱讀 38,432評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼胆建!你這毒婦竟也來(lái)了烤低?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,088評(píng)論 0 261
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤笆载,失蹤者是張志新(化名)和其女友劉穎扑馁,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體凉驻,經(jīng)...
    沈念sama閱讀 43,586評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡腻要,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了沿侈。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片闯第。...
    茶點(diǎn)故事閱讀 38,137評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖缀拭,靈堂內(nèi)的尸體忽然破棺而出咳短,到底是詐尸還是另有隱情,我是刑警寧澤蛛淋,帶...
    沈念sama閱讀 33,783評(píng)論 4 324
  • 正文 年R本政府宣布咙好,位于F島的核電站,受9級(jí)特大地震影響褐荷,放射性物質(zhì)發(fā)生泄漏勾效。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評(píng)論 3 307
  • 文/蒙蒙 一叛甫、第九天 我趴在偏房一處隱蔽的房頂上張望层宫。 院中可真熱鬧,春花似錦其监、人聲如沸萌腿。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,333評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)毁菱。三九已至米死,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間贮庞,已是汗流浹背峦筒。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,559評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留窗慎,地道東北人物喷。 一個(gè)月前我還...
    沈念sama閱讀 45,595評(píng)論 2 355
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像捉邢,于是被迫代替她去往敵國(guó)和親脯丝。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評(píng)論 2 345

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

  • Python 內(nèi)置的sorted的函數(shù)可以用來(lái)對(duì)自定義的數(shù)據(jù)結(jié)構(gòu)(列表)(設(shè)該列表為myList)排序伏伐,用法如下宠进。...
    Minda1987閱讀 4,754評(píng)論 0 50
  • 示例: 給定一堆數(shù)字 "3,31,34,2", 把它們排列成最大的數(shù)343312 題解:將字符串用逗號(hào)拆分,"3,...
    胖噠就是我閱讀 1,873評(píng)論 0 0
  • 定義: sorted() 函數(shù)對(duì)所有可迭代的對(duì)象進(jìn)行排序操作藐翎。 內(nèi)建函數(shù) sorted 方法返回的是一個(gè)新的 li...
    python追求者閱讀 468評(píng)論 0 0
  • 表情是什么材蹬,我認(rèn)為表情就是表現(xiàn)出來(lái)的情緒。表情可以傳達(dá)很多信息吝镣。高興了當(dāng)然就笑了堤器,難過(guò)就哭了。兩者是相互影響密不可...
    Persistenc_6aea閱讀 124,205評(píng)論 2 7
  • 16宿命:用概率思維提高你的勝算 以前的我是風(fēng)險(xiǎn)厭惡者末贾,不喜歡去冒險(xiǎn)闸溃,但是人生放棄了冒險(xiǎn),也就放棄了無(wú)數(shù)的可能拱撵。 ...
    yichen大刀閱讀 6,033評(píng)論 0 4