-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathSolution.cpp
86 lines (71 loc) · 2.67 KB
/
Solution.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
// https://leetcode.com/problems/invalid-transactions/
class Solution {
public:
typedef tuple<string, int, int, string, int> transaction;
static string get_str(string str, int& pos) {
string ret;
while (isalpha(str[pos])) ret += str[pos++];
if (str[pos]) ++pos;
return ret;
}
static int get_int(string str, int& pos) {
int ret{};
while (isdigit(str[pos])) ret = ret * 10 + str[pos++] - '0';
if (str[pos]) ++pos;
return ret;
}
static transaction parse(const std::string& str, int id) {
int pos = 0;
string name = get_str(str, pos);
int time = get_int(str, pos);
int value = get_int(str, pos);
string city = get_str(str, pos);
return {name, time, value, city, id};
}
vector<string> invalidTransactions(vector<string>& transactions) {
int n = transactions.size();
vector<transaction> parsed(n);
for (int i = 0; i < n; ++i) parsed[i] = parse(transactions[i], i);
sort(parsed.begin(), parsed.end());
// To do range updates using prefix sum
// When we identify a range [j, i] as bad,
// increment badness[j] and decrement badness[i+1]
// On taking prefix sum, this will increment every element in [j, i] by 1
vector<int> badness(n + 1);
// Active cities for current user:
// Where a transaction occured in last 60 minutes
// map: city -> count of transactions
unordered_map<string, int> active_cities;
// Sliding window from j to i:
// active transactions (last 60 minutes) for current user
for (int i = 0, j = 0; i < n; ++i) {
const auto& [name, time, value, city, id] = parsed[i];
// If the user changed, reset active city
if (i and name != get<0>(parsed[i - 1])) {
j = i;
active_cities.clear();
}
++active_cities[city];
// If value is above threshold, [i, i] is a bad range
if (value > 1000) ++badness[i], --badness[i + 1];
// Sliding window update:
// Decrement count of cities if they go outside active window
// If count becomes 0, remove it from hashmap
while (get<1>(parsed[j]) < time - 60) {
const auto& prevcity = get<3>(parsed[j]);
if (--active_cities[prevcity] == 0) active_cities.erase(prevcity);
++j;
}
// If active city count > 1, range of transactions [j, i] are all invalid
if (active_cities.size() > 1) ++badness[j], --badness[i + 1];
}
vector<string> ret;
// Take prefix sum of badness array
// Transaction is invalid iff prefix sum > 0
for (int i = 0, badness_pref = 0; i < n; ++i) {
if (badness_pref += badness[i])
ret.push_back(transactions[get<4>(parsed[i])]);
}
return ret;
}
};