Python3 sort 自定义排序逻辑:lambda函数、可比较类、cmp_to_key

Python3 sort 自定义排序逻辑:lambda函数、可比较类、cmp_to_key

以字符串排序为例,现在我们有这样一个字符串的 list :

words = ["vanilla", "tom", "sue", "david", "bill", "tom", "jezebel", "jackb", "jack", "jacka"]

默认排序方式

对于字符串而言,其默认的比较大小逻辑为:按首个字符顺序比较大小,相同则看后一个字符,后一个字符为空者较小。

def test1(things):
    things.sort()
    print(things)
    
test1(words.copy())

输出:

['bill', 'david', 'jack', 'jacka', 'jackb', 'jezebel', 'sue', 'tom', 'tom', 'vanilla']

lambda函数

sort 方法可以接收 key 参数,传递一个如下的可调用的对象:
以下例子通过 lambda 函数设置新的排序逻辑:以字符串长度为基准。
lambda 函数需要:1.接收1个参数;2.返回一个值,且返回值是可比较大小的。

def test2(things):
    things.sort(key=lambda e: len(e))
    print(things)

test2(words.copy())

输出:

['tom', 'sue', 'tom', 'bill', 'jack', 'david', 'jackb', 'jacka', 'vanilla', 'jezebel']

自定义可比较类

但是 lambda 函数无法表示比较复杂的排序逻辑。比如我们想:优先按照字符串长度为基准排序,再按照字母大小排序。lambda 函数是做不到如此复杂的逻辑的。
实际上, key 参数也可以传接收1个参数以初始化的类,比如下面的继承 str 类的 Cmp 类(因为 str 本身就接收一个参数以初始化,所以无需重写 Cmp__init__ 函数)
为了使 Cmp 是可比较大小的,需定义用于比较的 magic method,这些函数为: __gt__, __lt__,__ge__, __le__, __eq__,分别在进行 >, <, >=, <=, == 运算时被调用,一个鲁棒的可比较类需要包含全部这五个函数的定义。
Cmp 中只重写了 __lt__ ,因为在 sort 的执行中,优先进行 < 运算(只有在 < 运算抛出异常的情况下,才会试图进行其它比较运算)

def test3(things):
    class Cmp(str):
        def __lt__(self, other):
            if len(self) == len(other):
                return str.__lt__(self, other)
            else:
                return len(self) < len(other)
    things.sort(key=Cmp)
    print(things)

test3(words.copy())

输出:

['sue', 'tom', 'tom', 'bill', 'jack', 'david', 'jacka', 'jackb', 'jezebel', 'vanilla']

通过cmp_to_key自定义比较逻辑

定义鲁棒的 Cmp 类比较麻烦,毕竟两个对象的大小比较,无外乎“大于”、“小于”、“等于”三个状态。最直接的方法难道不是定义一个函数,输入两个参数,输出前者与后者之间的大小关系吗?可是有这样的接口吗?有的。
通过 functools 模块里的 cmp_to_key 函数,可以简洁地将上述自定义函数转化成 key 可接收的格式。
自定义函数需要:1.接收两个参数 p1, p2;2.返回1、0或-1,其中1代表 p1 > p2,0代表 p1 == p2, -1代表 p1 < p2

def test4(things):
    def compare(s1, s2):
        if len(s1) == len(s2):
            for c1, c2 in zip(s1, s2):
                if c1 > c2:
                    return 1
                elif c1 < c2:
                    return -1
            return 0  # 这里的比较其实可以直接使用字符串之间的 >, <, == 实现,之所以手动实现,是为了强调,没有内置比较函数的类也可以手动实现自定义的比较
        else:
            if len(s1) > len(s2):
                return 1
            else:
                return -1
    from functools import cmp_to_key
    things.sort(key=cmp_to_key(compare))
    print(things)

test4(words.copy())

输出:

['sue', 'tom', 'tom', 'bill', 'jack', 'david', 'jacka', 'jackb', 'jezebel', 'vanilla']
  • 5
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Python中,我们可以通过自定义排序函数来实现对列表或字典的自定义排序。首先,如果我们想对字典按照值进行排序,可以使用sorted函数lambda表达式来实现。例如,我们可以按照值的降序排列一个字典s: s = {'a': 45, 'b': 44, 'c': 78, 'd': 34} a = sorted(s.items(), key=lambda x: x, reverse=True) print('按照value降序排序:', a) 同样地,我们也可以按照值的升序排列字典s: a1 = sorted(s.items(), key=lambda x: x, reverse=False) print('按照value升序排序:', a1) 如果我们想对列表按照某个元素进行排序,可以使用sort方法或sorted函数,并通过自定义排序函数来指定排序依据。例如,假设我们要对列表a进行按照元素mode 7后的升序排序,可以定义一个compare_personal函数来实现: import functools a = [3, 8, 14, 6, 7] def compare_personal(x, y): return x % 7 - y % 7 a.sort(key=functools.cmp_to_key(compare_personal)) b = sorted(a, key=functools.cmp_to_key(compare_personal)) print(a) print(b) 这样,我们就可以得到按照元素mode 7后的升序排列的列表a和b: [14, 7, 8, 3, 6]<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [python 自定义排序](https://blog.csdn.net/dpengwang/article/details/85839652)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [一文教你搞定Python如何自定义标准排序](https://blog.csdn.net/weixin_57662182/article/details/127890324)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值