1751. Maximum Number of Events That Can Be Attended II
Problem Description
In this problem, you are provided with an array named events
, where each element in this array is another array that contains three integers: [startDay_i, endDay_i, value_i]
. The i-th
element represents an event that begins on startDay_i
, ends on endDay_i
, and attending this event yields a reward of value_i
. Additionally, an integer k
is given, indicating the maximum number of events you can attend.
However, there are constraints: you can attend only one event at a time, and if you attend an event, you must be present for the entire duration of the event, from startDay_i
to endDay_i
inclusive. This means that you cannot attend part of an event, and you also can't jump into another event that overlaps in date with the current one.
The goal is to select events to attend in such a way that maximizes the sum of values received, without exceeding the max number of events (k
) that you can attend. You must then return the maximum sum of values you can achieve.
Intuition
To solve this problem, a dynamic programming approach is used. The events are first sorted based on their end dates to facilitate sequential processing from the earliest end date to the latest.
The core of the approach is to construct a 2D array f
, where f[i][j]
represents the maximum value that can be obtained by considering up to the i-th
event and attending at most j
events. By iterating through each event (denoted as i
), we have the option to either attend this event or not. If the current event is attended, we add its value to the maximum value obtained by attending j-1
events before the current event's start. In case the current event is not attended, we carry forward the maximum value of attending j
events without considering the current event.
To check which other events could have been attended before the i-th
event, a binary search is performed to find the rightmost event that ends before the current event's start, ensuring no overlap. This efficient search is possible due to the initial sorting of the events by their end dates.
Finally, the resultant value f[n][k]
(where n
is the number of events) will give us the maximum sum we can achieve by attending up to k
events. The solution leverages the Python bisect_left
function for binary search, and iteratively updates the dynamic programming table to arrive at the solution.
Learn more about Binary Search, Dynamic Programming and Sorting patterns.
Solution Approach
The solution is implemented using Dynamic Programming (DP), which is a method for solving complex problems by breaking them down into simpler subproblems. It relies on the fact that the optimal solution to a problem depends upon the optimal solution to its subproblems.
Let's walk through the code:
-
Sort
events
by their end time to process them in order of completion.events.sort(key=lambda x: x[1])
-
Initialize the DP table
f
withn + 1
rows andk + 1
columns filled with zeros, wheren
is the number of events. The DP table is used to store the maximum value that can be obtained by attending a certain number of events up to a certain point in time.f = [[0] * (k + 1) for _ in range(n + 1)]
-
Loop over sorted events, with each event enumerated as
i
, starting from 1. This loop is responsible for filling the DP table with the maximum values for each scenario of attending up toj
events when you are considering up to thei-th
event.for i, (st, _, val) in enumerate(events, 1):
-
Inside the loop, for each event, use binary search to find the rightmost event that ends before the current event starts (to ensure no overlap) using
bisect_left
.p = bisect_left(events, st, hi=i - 1, key=lambda x: x[1])
-
Nested within is another loop over the range
1
tok + 1
(the maximum number of events you can attend), denoted asj
. For each possible number of events to attend, we are calculating the maximum value possible either by including thei-th
event or excluding it.for j in range(1, k + 1): f[i][j] = max(f[i - 1][j], f[p][j - 1] + val)
Here's an explanation of each case:
-
f[i - 1][j]
represents the value obtained by not attending thei-th
event, hence just carrying over the value from the previous event iteration. -
f[p][j - 1] + val
represents the value obtained by attending thei-th
event. The value from the event right before the current event starts (f[p][j - 1]
) is added to the current event's value (val
).
-
-
Return the value
f[n][k]
, which is the answer to the problem. It represents the maximum sum that can be achieved by attending up tok
events.return f[n][k]
In this implementation, the data structure used is a 2D array/list for DP. Also, a binary search pattern is used for efficient searching of compatible events.
This DP approach ensures we are considering every combination of events we could possibly attend (up to k
events) and updating our max sum at each step for each event choice. By doing this iteratively and considering all choices, we ensure that the result will contain the maximum sum that can be achieved under the given conditions.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's illustrate the solution approach with a small example.
Suppose events = [[1, 2, 4], [2, 3, 3], [3, 4, 2]]
and k = 2
. We're allowed to attend up to 2 events to maximize the sum of the values.
Step 1: Sort the events by their end time. After sorting, the events
array remains [[1, 2, 4], [2, 3, 3], [3, 4, 2]]
.
Step 2: Initialize the DP table f
with rows equivalent to the number of events n + 1
and columns k + 1
.
f = [ [0, 0, 0], # n=0 events [0, 0, 0], # n=1 event [0, 0, 0], # n=2 events [0, 0, 0] # n=3 events ]
Step 3: Begin looping over sorted events and enumerate the events starting from 1:
- The events are:
- Event 0: [1, 2, 4]
- Event 1: [2, 3, 3]
- Event 2: [3, 4, 2]
Step 4: Use binary search to find the rightmost event that does not conflict with the current one. First, we check for each event manually.
- For Event 0, we look before event 0: no events.
- For Event 1, we look before event 1: event 0 ends before event 1 starts.
p = 0
- For Event 2, we look before event 2: event 1 ends as event 2 starts, so only event 0 does not conflict.
p = 0
Step 5: Inside the loop for i
(events), loop for j
(number of events to attend up to k
). For each j
, we compute the max
of not attending the current event or attending it:
- For Event 0:
j = 1: f[1][1] = max(f[0][1], f[0][0] + 4)
becomes4
j = 2: f[1][2] = max(f[0][2], f[0][1] + 4)
becomes4
- For Event 1, considering previous values:
j = 1: f[2][1] = max(f[1][1], f[0][0] + 3)
stays4
j = 2: f[2][2] = max(f[1][2], f[0][1] + 3)
becomes7
(Event 0 + Event 1)
- For Event 2, considering previous values:
j = 1: f[3][1] = max(f[2][1], f[0][0] + 2)
stays4
j = 2: f[3][2] = max(f[2][2], f[0][1] + 2)
stays7
(cannot attend Event 2 after Event 1 due to overlap)
The updated f
table after processing would look like:
f = [ [0, 0, 0], # n=0 events [0, 4, 4], # n=1 event, up to k=2 [0, 4, 7], # n=2 events, up to k=2 [0, 4, 7] # n=3 events, up to k=2 ]
Step 6: Return f[n][k]
which is 7
, representing the maximum sum of values by attending up to k=2
events.
Note that we were able to include the values of only the first two events, ignoring the third event since attending would overlap with the second event. This gives us the maximum value of 7
for attending up to two events.
Solution Implementation
1from bisect import bisect_left
2from typing import List
3
4class Solution:
5 def max_value(self, events: List[List[int]], k: int) -> int:
6 # First, we sort the events based on their ending time.
7 events.sort(key=lambda event: event[1])
8
9 # The number of events
10 num_events = len(events)
11
12 # Prepare a DP table with dimensions (number of events + 1) x (k + 1)
13 dp_table = [[0] * (k + 1) for _ in range(num_events + 1)]
14
15 # Iterate through each event, starting with index 1 for convenience
16 for i, (start_time, _, value) in enumerate(events, 1):
17 # Find the index of the last event that does not overlap with the current event
18 previous_index = bisect_left(events, start_time, hi=i - 1, key=lambda event: event[1])
19
20 # Update DP table for each of the k attendances
21 for j in range(1, k + 1):
22 # The value is the max between not attending this event and attending it
23 dp_table[i][j] = max(
24 dp_table[i - 1][j], # Not picking current event
25 dp_table[previous_index][j - 1] + value # Picking current event
26 )
27
28 # The maximum value for attending at most k events is stored at dp_table[num_events][k]
29 return dp_table[num_events][k]
30
31# The function `max_value` can be called with a list of events and a number k to find the result.
32
1class Solution {
2 public int maxValue(int[][] events, int k) {
3 // Sort the events array by the end time of each event.
4 Arrays.sort(events, (a, b) -> a[1] - b[1]);
5 int n = events.length;
6
7 // Create a DP table where f[i][j] will hold the maximum value for the first i events using j events.
8 int[][] dpTable = new int[n + 1][k + 1];
9
10 // Fill in the DP table
11 for (int i = 1; i <= n; ++i) {
12 // Start time and value of the current event
13 int startTime = events[i - 1][0];
14 int value = events[i - 1][2];
15
16 // Find the previous event that does not overlap with the current event
17 int previous = binarySearch(events, startTime, i - 1);
18
19 for (int j = 1; j <= k; ++j) {
20 // The value for using j events including the i-th event is either:
21 // 1. The same as using the first i-1 events (not using the i-th event), or
22 // 2. The value of using j-1 events plus the value of the i-th event
23 // (when j-1 events are used up to the previously non-overlapping event).
24 dpTable[i][j] = Math.max(dpTable[i - 1][j], dpTable[previous][j - 1] + value);
25 }
26 }
27
28 // Return the maximum value achievable using k events
29 return dpTable[n][k];
30 }
31
32 // Performs a binary search to find the index of the latest event that finishes before 'endTime'.
33 private int binarySearch(int[][] events, int endTime, int high) {
34 int low = 0;
35 while (low < high) {
36 int mid = (low + high) >> 1; // Equivalent to (low + high) / 2, but avoids overflow
37 if (events[mid][1] >= endTime) {
38 high = mid;
39 } else {
40 low = mid + 1;
41 }
42 }
43 return low;
44 }
45}
46
1#include <vector>
2#include <algorithm>
3#include <cstring>
4
5class Solution {
6public:
7 int maxValue(vector<vector<int>>& events, int maxEvents) {
8 // Sort the events based on their end time
9 sort(events.begin(), events.end(), [](const auto& event1, const auto& event2) {
10 return event1[1] < event2[1];
11 });
12
13 int numEvents = events.size();
14 int dp[numEvents + 1][maxEvents + 1];
15
16 // Initialize the dp array with zero
17 memset(dp, 0, sizeof(dp));
18
19 // Loop through all events
20 for (int i = 1; i <= numEvents; ++i) {
21 // Get the start time and value of the current event
22 int startTime = events[i - 1][0];
23 int value = events[i - 1][2];
24
25 // Create a vector that holds the start time
26 vector<int> currentTime = {startTime};
27
28 // Find the last event that finishes before the current event starts
29 int lastEventIndex = lower_bound(events.begin(), events.begin() + i - 1, currentTime, [](const auto& a, const auto& b) {
30 return a[1] < b[0];
31 }) - events.begin();
32
33 // Loop through the numbers 1 to maxEvents
34 for (int j = 1; j <= maxEvents; ++j) {
35 // Update the dp array with the max value achieved by either
36 // skipping the current event or attending it after the last non-overlapping event
37 dp[i][j] = max(dp[i - 1][j], dp[lastEventIndex][j - 1] + value);
38 }
39 }
40
41 // Return the maximum value that can be achieved by attending maxEvents number of events
42 return dp[numEvents][maxEvents];
43 }
44};
45
1function maxValue(events: number[][], maxEvents: number): number {
2 // Sort the events based on their end time
3 events.sort((a, b) => a[1] - b[1]);
4
5 // Number of events
6 const numberOfEvents = events.length;
7
8 // Initialize the dynamic programming table with 0's
9 const dp: number[][] = new Array(numberOfEvents + 1)
10 .fill(0)
11 .map(() => new Array(maxEvents + 1).fill(0));
12
13 // Function to search for the latest event that does not overlap
14 const binarySearchLatestNonOverlappingEvent = (startTime: number, high: number): number => {
15 let low = 0;
16 let mid;
17 while (low < high) {
18 mid = (low + high) >> 1;
19 if (events[mid][1] >= startTime) {
20 high = mid;
21 } else {
22 low = mid + 1;
23 }
24 }
25 return low;
26 };
27
28 // Iterate over each event
29 for (let i = 1; i <= numberOfEvents; ++i) {
30 // Destructure the event details
31 const [start, _, value] = events[i - 1];
32
33 // Find the latest event that does not overlap
34 const latestNonOverlappingEventIndex = binarySearchLatestNonOverlappingEvent(start, i - 1);
35
36 // Update dynamic programming table
37 for (let j = 1; j <= maxEvents; ++j) {
38 dp[i][j] = Math.max(dp[i - 1][j], dp[latestNonOverlappingEventIndex][j - 1] + value);
39 }
40 }
41
42 // Return the maximum value that can be obtained by attending at most 'maxEvents' events
43 return dp[numberOfEvents][maxEvents];
44}
45
Time and Space Complexity
The given code is a dynamic programming solution that solves a problem similar to the weighted job scheduling problem. The solution involves finding the maximum value that can be obtained by attending up to k
non-overlapping events from a list of events given their start time, end time, and value.
Time Complexity
The time complexity of the provided code is determined by two main operations: sorting the events and populating the f
table.
-
Sorting: The events list is sorted based on the end time using the
sort
function. If there aren
events, the time complexity for sorting isO(n log n)
. -
DP Table Calculation:
- For each event
i
(from1
ton
), we perform abisect_left
which takesO(log i)
time per event. - Then for each event
i
, we iterate throughj
from1
tok
, resulting inO(k)
operations per event. - Therefore, for all
n
events, we will haveO(nk log n)
work inside the loop for updating thef
table.
- For each event
Combining both steps, the total time complexity becomes O(n log n) + O(nk log n)
. Since both terms dominate for different ranges of n
and k
, we usually express this as O(nk log n)
as it's the term that will be larger in the worst-case scenario (where k
is large enough).
Space Complexity
The space complexity of the code is mainly determined by the size of the f
table which is a two-dimensional list in Python.
- We have
n + 1
rows andk + 1
columns in thef
table ([[0] * (k + 1) for _ in range(n + 1)]
), thus the space complexity isO(nk)
.
To summarize, the time complexity is O(nk log n)
and the space complexity is O(nk)
.
Learn more about how to find time and space complexity quickly using problem constraints.
Which two pointer techniques do you use to check if a string is a palindrome?
Recommended Readings
Binary Search Speedrun For each of the Speedrun questions you will be given a binary search related problem and a corresponding multiple choice question The multiple choice questions are related to the techniques and template s introduced in the binary search section It's recommended that you have gone through at
What is Dynamic Programming Prerequisite DFS problems dfs_intro Backtracking problems backtracking Memoization problems memoization_intro Pruning problems backtracking_pruning Dynamic programming is an algorithmic optimization technique that breaks down a complicated problem into smaller overlapping sub problems in a recursive manner and uses solutions to the sub problems to construct a solution
Sorting Summary Comparisons We presented quite a few sorting algorithms and it is essential to know the advantages and disadvantages of each one The basic algorithms are easy to visualize and easy to learn for beginner programmers because of their simplicity As such they will suffice if you don't know any advanced
Want a Structured Path to Master System Design Too? Don’t Miss This!