不跳过数字的Python排名密集排名

问题描述

使用 python 3.x 我想实现密集排名(如果排名重复,不要跳过数字)。我有以下数组根据分数排序

rank_list = [{'Id': 236966,'score': 91.0},{'Id': 237241,'score': 82.0},{'Id': 237077,'score': 79.0},{'Id': 237084,'score': 78.0},{'Id': 237080,'score': 72.0},{'Id': 237236,'score': 71.0},{'Id': 236979,{'Id': 236909,'score': 67.0},{'Id': 237174,{'Id': 237035,'score': 66.0}] 

我使用下面的代码来计算和分配“排名”字段,但如果排名重复,代码会跳过排名

def rankFunc(e):
    return e['score']

rank_list.sort(key = rankFunc,reverse=True)
sorted_scores = [obj['score'] for obj in rank_list]
ranks = [sorted_scores.index(x) for x in sorted_scores]
for index,obj in enumerate(rank_list):
    obj['rank'] = ranks[index]+1

当前输出

等级 = [1,2,3,4,5,6,8,10]

我希望在不跳过任何数字的情况下分配等级,

等级 = [1,7,8]

解决方法

这个小辅助函数应该可以解决您的问题:

def rank_unique(x,**kwargs):
    sx = sorted(set(x),**kwargs)
    invsx = {s: i for i,s in enumerate(sx)}
    return [1 + invsx[v] for v in x]

>>> rank_unique([r["score"] for r in rank_list],reverse=True)
[1,2,3,4,5,6,7,8]
,

这个简单的解决方案将解决您的问题(假设数组已经排序)。

def assignRank(_list):
    rank = 0
    for i in range(len(_list)):
        if _list[i]["score"] != _list[i-1]["score"]:
            rank += 1
        _list[i]["rank"] = rank
,

你可以使用带有pandas函数的dense rank方法

import pandas as pd
rank = pd.Series(sorted_scores).rank(method="dense",ascending=False).values
print(rank)

输出:

[1. 2. 3. 4. 5. 6. 6. 7. 7. 8.]

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...