问题描述
我的 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
)