-
Notifications
You must be signed in to change notification settings - Fork 363
/
Copy pathweighted_activity_selection.hpp
131 lines (110 loc) · 3.32 KB
/
weighted_activity_selection.hpp
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
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
/*
Weighted activity selection problem
--------------------
Given N activities where every activity is represented by
following three elements of it.
1. Start Time
2. Finish Time
3. Weight or Value Associated
Find the maximum weight subset of activities such that no two of them in the subset overlap.
Time complexity
---------------
O(N*lg(N))
Space complexity
----------------
O(N)
Author
------
Nikolaus Fedurko (@B1Z0N)
*/
#ifndef WEIGHTED_ACTIVITY_SELECTION_HPP
#define WEIGHTED_ACTIVITY_SELECTION_HPP
#include <ios>
#include <vector>
#include <algorithm>
#include <ctime>
/**
* Struct for single activity data
*/
struct Activity
{
std::time_t start;
std::time_t end;
std::size_t weight;
};
/**
* Find the index of right-most
* non-overlaping activity
* that is left to the query[ index ]
* using binary search in O( lg(index) )
*/
int left_activity_bsearch( const std::vector<Activity>& quer, std::size_t index )
{
int lo = 0, hi = index - 1;
while ( lo <= hi ) {
int mid = ( lo + hi ) / 2;
// if it is non-overlapping
if ( quer[ mid ].end <= quer[ index ].start ) {
// if there are other non-overlapping activity
// a bit to the right, then continue searching
if ( quer[ mid + 1 ].end <= quer[ index ].start ) {
lo = mid + 1;
}
// if it is right-most non-overlapping
else {
return mid;
}
}
// if there are overlaps, in the middle
// look to the left half
else {
hi = mid - 1;
}
}
// if quer[ index ] has no non-overlapping activities
// to the left of it
return -1;
}
/**
* Algorithm of solution
*/
int weighted_activity(const std::vector<std::time_t>& start,
const std::vector<std::time_t>& end,
const std::vector<std::size_t>& weight) {
std::vector<Activity> quer;
for ( std::size_t i = 0; i < start.size(); i++ ) {
quer.push_back( { start[ i ], end[ i ], weight[ i ] } );
}
// sort by end in ascending order
std::sort( std::begin( quer ), std::end( quer ),
[] (const Activity& fst, const Activity& snd)
{
return fst.end < snd.end;
}
);
// sol[ i ] stores solution to first i + 1 activities
std::vector<std::size_t> sol ( quer.size() );
// first solution is just it's single weight
sol[0] = quer[0].weight;
// find all solutions
for ( std::size_t i = 1; i < quer.size(); i++ ) {
std::size_t weight_with_current = quer[i].weight;
// j - index of the problem needed to solve if
// we want to include our i index in solution
int j = left_activity_bsearch( quer, i );
// if there are consistent j on the left,
// add max_weight of it's solution
if ( j != -1 ) {
weight_with_current += sol[j];
}
// decide whether to include this i index into solution
// depending on it's weights
sol[ i ] = std::max(
sol[ i - 1 ], // weight without current
weight_with_current
);
}
// last index contains solution to whole problem
return sol.back();
}
#endif // WEIGHTED_ACTIVITY_SELECTION_HPP