-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtiedrank.h
89 lines (71 loc) · 2.23 KB
/
tiedrank.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
/*=================================================================
*
* tiedrank.h
* Author: Andrew Magis
* Calculate tiedrank on vector of data. Output vector is
* passed in by reference. All tiedranks are averaged.
*
* Inputs: reference to vector of floats
* Outputs: void
*=================================================================*/
#include <vector>
#include <algorithm>
void tiedrank(std::vector<float> &data, std::vector<float> &ranks) {
//Clear the ranks, in case it has anything in it
ranks.clear();
ranks.resize(data.size(), 0);
//Vector of pairs to store the indices
std::vector< std::pair<float, int> > col;
//Inner loop, looping over every row
for (int j = 0; j < data.size(); j++) {
//Copy the element into the pair template with index
std::pair<float, int> element(data[j], j);
//Add this pair to the vector
col.push_back(element);
}
//Sort the vector using quicksort
std::sort(col.begin(), col.end());
std::vector<float> temp;
int crank = 1, tied = 1;
float average = 1.f;
temp.push_back(crank);
for (int j = 1; j < col.size(); j++) {
//Compare this one to the previous one
if (col[j].first == col[j-1].first) {
tied++;
average += ++crank;
temp.push_back(-1);
} else {
//If they are not equal, and tied == 0, do nothing but add in
//the current rank
if (tied == 1) {
average = ++crank;
temp.push_back(crank);
//Otherwise, go back through and average the ranks for all tied elements
} else {
average /= (float)tied;
for (int k = temp.size()-tied; k <= temp.size()-1; k++) {
temp[k] = average;
}
average = (float)++crank;
temp.push_back(crank);
tied = 1;
}
}
}
//Check to make sure the final one is not tied
if (tied > 1) {
average /= (float)tied;
for (int k = temp.size()-tied; k <= temp.size()-1; k++) {
temp[k] = average;
}
}
/*
for (int j = 0; j < col.size(); j++ ) {
printf("%d: %.3f %d %.3f\n", j, col[j].first, col[j].second, temp[j]);
}*/
//Finally, put the ranks into the output vector using the original indices
for (int j = 0; j < col.size(); j++) {
ranks[col[j].second] = temp[j];
}
}