forked from markhary/codility
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathEquiLeader.cpp
105 lines (92 loc) · 3.12 KB
/
EquiLeader.cpp
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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
// https://app.codility.com/programmers/lessons/8-leader/equi_leader/
//
// Task Score: 100%
// Correctness: 100%
// Performance: 100% or Not assessed
// Detected time complexity: O(N)
//
#include <vector>
#include <stack>
using namespace std;
// Expected time complexity is O(N)
// Expected space complexity is O(N)
int solution(vector<int> &A) {
const int N = A.size();
// First, L_l = L_r == L
// Algorithm:
// - Iterate and find leader
// - Now that we know leader, go back through and iterate again,
// finding equileaders. We know N and L and our current index, and
// can do the math
stack<int> candidates;
// Find the leader using the algorithm suggested by codility
// This is O(N)
for (int i=0; i<N; i++) {
if ( !candidates.empty() ) {
// If the top isn't equal to this, remove them both
int candidate = candidates.top();
if ( candidate != A[i] ) {
candidates.pop();
} else {
candidates.push(A[i]);
}
} else {
// empty, so push this on the list
candidates.push(A[i]);
}
}
// I have my candidate. If list is empty, then there are no candidates
// and we can return. If list is not empty, candidate is in the list (they must
// all be the same)
if ( candidates.empty() ) {
return 0;
}
int candidate = candidates.top();
// Iterate through list again and determine if candidate is the leader
// Can't exit list early, because we have to know total number of leader
// elements in A
// This is O(N)
int count = 0;
for (int i=0; i<N; i++) {
if ( A[i] == candidate ) {
count++;
}
}
// Have to have > N/2 instances to be a leader
double half = ((double) N)/2;
if ( !((double (count)) > half) ) {
return 0;
}
int leader = candidate;
// Now know leader, so figure out how many equileaders there are
int numEquiLeaders = 0;
// An equileader is when:
// N is total length
// X = length of elements on left
// Y = length of elements on right (Y = N-X)
// - Number of leader elements in X is > X.size/2 AND
// - Number of leader elements in Y is > Y.size/2
// o Number of leader elements in Y is N - N_in_X
// Should manage itself when there are 0 elements in either X or Y
// This is O(N)
int countX = 0;
for (int i=0; i<N; i++) {
// Number of times leader found in X
if ( A[i] == leader ) {
countX++;
}
// number of times leader must be in Y
int countY = count - countX;
// Calculate sizes of X and Y
int sizeX = (i+1);
int sizeY = N-sizeX;
double halfSizeX = ((double) sizeX)/2;
double halfSizeY = ((double) sizeY)/2;
// Now check that leader is leader of both X and Y,
// if so increment equiLeaders
if ( (countX > halfSizeX ) && (countY > halfSizeY)) {
numEquiLeaders++;
}
}
return numEquiLeaders;
}