Laravel 中的积分排名,其中积分相等

问题描述

我正在研究如何更正显示积分排名以显示排行榜

我的 Modal 中目前有以下内容

 public function getPointrankAttribute(){
    $year = Carbon::parse(Carbon::Now())->year;

    $pointrank = Points::query()
        ->select('member_id')->selectRaw('SUM(value) as TotalPoints')
        ->where('Year','=',$year)
        ->groupBy('member_id')
        ->orderByDesc('Totalpoints')
        ->get();

    return $pointrank->search(function($points){
        return $points->member_id == $this->id;
    }) + 1;

}

这很好用,但是它使用表格中的行位置来显示排名,所以第一条记录是第一条,第二条记录显示为第二条。

然而,当成员的总分相同时会出现问题,因为在此情况下,排名基于表中的位置

看下面的例子

|member_id|totalpoints|
|   12    | 20        |
|    3    |  10       |
|   10    |  10       |
|    5    |   5       |

在这个例子中基于我当前的解决方

第 12 名成员 = 第 1 名,第 3 名成员 = 第 2 名,第 10 名成员 = 第 3 名,第 5 名成员 = 第 4 名

我想要的是以下

第 12 名成员 = 第 1 名,第 3 名成员 = 第 2 名,第 10 名成员 = 第 2 名,第 5 名成员 = 第 4 名

当我们有超过 1 个成员共享该位置时(如本例中的成员 3 和 10),如果我也可以添加“=”,那就太棒了

提前致谢

解决方法

要为您的用户获得排名,您需要先将排名分配给不同的总分,使用 Laravel 集合助手,您可以创建单独的排名集合,而无需考虑成员 ID,例如

$collection =collect(
[
  [
    "member_id" => 12,"totalpoints" => 20
  ],[
    "member_id" => 3,"totalpoints" => 10
  ],[
    "member_id" => 10,[
    "member_id" => 5,"totalpoints" => 5
  ],]
);

$ranks = $collection->unique('totalpoints')
                    ->values()
                    ->mapWithKeys(function ($item,$index) {
                        return [$item['totalpoints'] => $index + 1];
                    });

确保应根据总分对原始收藏品进行排序。

上面将创建一个不同点的数组作为键和值作为基于总点数的排名

Array
(
    [20] => 1
    [10] => 2
    [5] => 3
)

现在您可以根据原始集合中的 id 搜索您的成员,并从 ranks 数组中获取他们各自的排名

$id = 10;
$ranks[$collection->firstWhere("member_id",$id)["totalpoints"]];

如果供应商数据库支持 RANK() 等窗口函数,还有另一种方法可以直接从数据库中获取记录的排名

我想要的是以下内容:成员 12 = 1,成员 3 = 2,成员 10 = 2,成员 5 = 4

要获得您想要的结果,您只需从上面的代码中删除 ->values(),这样您的结果就不会被重新索引

$ranks = $collection->unique('totalpoints')
                    ->mapWithKeys(function ($item,$index) {
                        return [$item['totalpoints'] => $index + 1];
                    });

Array
(
    [20] => 1
    [10] => 2
    [5] => 4
)

DEMO