4

我正在尝试对形式为的字典进行排序:

d = {'+A':234, '-B':212, 'A':454, '-C':991, '-A':124}

我想按键对其进行排序,使其格式为:

+A, A, -A, +B, B, -B, etc

我一直在尝试使用sorted(d, key=lambda x: (x[1], x[0]) if len(x) == 2 else x[0]) ,但我似乎找不到任何方法对符号进行正确排序,因为它们在 ascii 图表上的顺序不正确。我究竟做错了什么?

4

3 回答 3

7

一种简单的方法

rank = ['+A', 'A', '-A', '+B', 'B', '-B', ...]
sorted(d.items(), key=lambda i: rank.index(i[0]))

如果有很多等级,最好使用adict

rank = {'+A': 0, 'A': 1, '-A': 2, '+B': 3, 'B': 4, '-B': 5, ...}
sorted(d.items(), key=lambda i: rank[i[0]])

您可以使用这样的 lambda 函数。请注意,使用后向切片以确保字母在其修饰符之前排序很重要。

sorted(d.items(), key=lambda i:(','+i[0])[::-1])

但我认为显式rank更清晰,不容易出现@Hari 答案中的错误。(到目前为止,有 5 人投了票而没有注意到这个错误)

如果您真的只需要对键进行排序(为什么?),您可以简单地使用rank.get而不是 lambda 函数:

>>> rank = {'+A': 0, 'A': 1, '-A': 2, '+B': 3, 'B': 4, '-B': 5, '+C': 6, 'C': 7, '-C': 8}
>>> d = {'+A':234, '-B':212, 'A':454, '-C':991, '-A':124}
>>> sorted(d, key=rank.get)
['+A', 'A', '-A', '-B', '-C']

但最好sorted完全跳过

>>> rank = ['+A', 'A', '-A', '+B', 'B', '-B', '+C', 'C', '-C']
>>> d = {'+A':234, '-B':212, 'A':454, '-C':991, '-A':124}
>>> [k for k in rank if k in d]
['+A', 'A', '-A', '-B', '-C']

如果你讨厌输入所有这些'

>>> rank = '+A A -A +B B -B +C C -C'.split()
于 2013-11-15T06:43:14.537 回答
6

这应该有效:

sorted(d, key=lambda x: (x[1], x[0]) if len(x) == 2 else (x[0], ','))

ascii 值介于和,之间,因此您可以在末尾放置一个虚拟对象进行比较。+-,

>>> d = {'+A':234, '-B':212, 'A':454, '-C':991, '-A':124, '+B':1, 'B':98, '+C':232, 'C':23}
>>> sorted(d, key=lambda x: (x[1], x[0]) if len(x) == 2 else (x[0], ','))
['+A', 'A', '-A', '+B', 'B', '-B', '+C', 'C', '-C']

您也可以简单地反转密钥并,为比较器附加 a:

sorted(d, key=lambda x: x[::-1] + ',')

所以 +A, A, -A 被比较为A+,,AA-,

于 2013-11-15T06:53:36.190 回答
1

我会排序:

  • 只是字母 - 去掉前导-+
  • +在(-1)上加上另一个权重,所以它首先出现,-所以它是最后一个 (1),否则为 0

例如:

sorted(d.iteritems(), key=lambda L: (L[0].lstrip('-+'), {'-': 1, '+': -1}.get(L[0][0], 0)))
# [('+A', 234), ('A', 454), ('-A', 124), ('-B', 212), ('-C', 991)]
于 2013-11-15T11:46:55.827 回答