php仿excel的rank函數也是借鑒網上的代碼,但是沒有二維數組情況下的進行rank排名,是以自己對代碼稍微改了一下,可以直接運作試驗。
<?php
$arr = array(
array('s'=>'99','r'=>'1','a'=>'a'),
array('s'=>'99','r'=>'1','b'=>'b'),
array('s'=>'100','r'=>'2','c'=>'c'),
array('s'=>'101','r'=>'3','d'=>'d')
);
echo '<pre>';
print_r(rank($arr,'s','r'));
//獲得一組數的名次的數組
function rank(array $array,$s,$r){
foreach($array as $k=>$v){
$marr[] = $v[$s];
}
foreach($array as $val){
$repeat=get_array_repeats($val[$s],$marr);
$num=gt_array_values($val[$s],$marr);
$rank[$r]=count($marr)-$num-$repeat+1;
$rank2[] = array_merge($val,$rank);
}
return $rank2;
}
//獲得比自己數小的個數
function gt_array_values($val,array $array){
$num=0;
for($i=0;$i<count($array);$i++){
if($val>$array[$i]){
$num++;
}
}
return $num;
}
//獲得這個數的重複次數
function get_array_repeats($string,array $array) {
$count = array_count_values($array);
foreach ($count as $key => $value) {
if ($key == $string) {
return $value;
}
}
}
邏輯也很簡單,學生的名次,等于總人數減去比自己成績低的人數,再減去和自己成績一樣的人數,再加一。
比自己成績低的人數,和與自己成績一樣多的人數,分别是兩個獨立的函數,前者需要循環所有學生成績,比自己成績低的進行人數累加,每個學生都要進行這個循環,有點耗性能哈。後者隻是單純引用數組函數解決問題。
這樣就可以根據考試成績(字段名s),得到考生的排名(字段名r),但是很明顯循環次數有點多,難以想象當資料量很大的時候情況是怎樣的。
如果不考慮二維數組的情況,有大神提出可以這樣寫:
$arr = array(195,180,180,161);
function rt($arr){
$temp = $arr;
rsort($temp);
$temp = array_unique($temp);
foreach ($arr as $k1 => $v1) {
foreach ($temp as $k2 => $v2) {
if($v1==$v2){
$res[$k1] = $k2+1;
}
}
}
return $res;
}
至于原理,說白了,rsort這個函數直接幫你把排名都排好了,鍵是排名,值是分數,是不是這個道理。然後循環原數組,分數一樣的,把這個排名取出來就行了。
至于這種方法,二維數組的時候要怎麼弄,這個需要二維數組裡的分數值先按順序排好,(這是另外一種算法),然後套用。
當然,這個方法理論上比上一個方法效率好多了。
另外查詢的時候還可以直接執行sql語句實作rank函數功能,這個網上有不少例子,如MySQL中rank函數如何實作
在這個例子中,當我們要查找id為1的學員的rank值的時候,直接運作sql語句如下:
select
tmp.id,tmp.name,tmp.score,
@j:=@j+1 as j,
@k:=(case when @pre_score=tmp.score then @k else @j end) as rank,
@pre_score:=tmp.score as pre_score
from
(
select * from score where id =1
) as tmp,
(select @k :=0,@j:=0, @pre_score:=0)
as mscore
@k:=(case when @pre_score=tmp.score then @k else @j end