881. Boats to Save People
Problem Description
In this problem, we are provided with an array called people
, where each element represents the weight of a person. We also have an infinite number of boats that can carry a maximum weight of limit
. Each boat can carry at most two people at the same time, as long as the total weight of these people doesn't exceed the limit
. Our objective is to determine the minimum number of boats required to carry every person.
Intuition
To solve this problem, we can use a two-pointer technique. The main idea is to pair the heaviest person left with the lightest person that can fit in the same boat with them, thus optimizing the usage of boat space. We start by sorting the people
array to be able to efficiently pair people.
After sorting, we set two pointers: one at the start (i
) and one at the end (j
) of the array. The i
pointer represents the lightest person left to place on a boat, while j
represents the heaviest. We loop until our two pointers meet. In each iteration, we try to place the heaviest person (people[j]
) in a new boat and see if we can also fit the lightest person (people[i]
) with them without exceeding the limit
.
- If we can fit both the
i
person and thej
person on the same boat, we increment thei
pointer (meaning thei
person is now on a boat). - Regardless of whether the
i
person got on the boat or not, we decrement thej
pointer, indicating that thej
person is now on a boat, and increment the counterans
, which tracks the number of boats used.
We continue this process until every person has been placed on a boat. The counter ans
gives us the minimum number of boats needed to carry everyone.
Learn more about Greedy, Two Pointers and Sorting patterns.
Solution Approach
The solution approach for this problem hinges on the efficient use of a two-pointer technique and the algorithm that supports it. Here's a walkthrough of the implementation details:
-
Sorting the Array: We begin by sorting the
people
array. Sorting is crucial because it allows us to consider the lightest and heaviest weights that can possibly be paired together. In Python, this is done with thesort()
method, which sorts the array in increasing order. Sorting is a common preprocessing step in many algorithms to facilitate ordered operations. -
Two-Pointer Technique: After sorting, we initialize two pointers:
i
, pointing to the start, andj
, pointing to the end of the array. These pointers will move towards each other as we pair up people into boats. The two-pointer technique is a common pattern used to solve problems that involve arrays, especially when looking for pairs that meet a certain criteria, as it is both space efficient and has a linear time complexity in this context. -
Iterating and Pairing: As we iterate through the array, we check if the person at the
i
-th position (lightest remaining) can be paired with the person at thej
-th position (heaviest remaining) without exceeding the weightlimit
.-
If the sum of weights
people[i] + people[j]
is less than or equal tolimit
, it means both can share one boat. We then move thei
pointer up (incrementing it by 1) to consider the next lightest person for the subsequent iterations. -
The
j
pointer is moved down (decremented by 1) after every iteration since thej
-th person is always placed in a boat (alone if not with thei
-th person).
-
-
Counting Boats: We have a counter,
ans
, which is incremented in every iteration of the loop. This counter represents the number of boats used. -
Loop Termination: The loop continues until
i > j
, which means all persons have been paired up and placed on boats. At this point, the value ofans
represents the minimum number of boats required to carry all persons.
The algorithm uses no additional data structures and operates directly on the input array. Therefore, the space complexity is O(1), as it only uses a fixed amount of extra space. The time complexity, after sorting, is O(n), where n is the number of people, since every person is considered exactly once when pairing them.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let us consider a small example to illustrate the solution approach. Suppose we have an array people
given as [3, 5, 3, 4]
and the limit
of the boats is 5
. We want to find out the minimum number of boats needed to carry everyone following the solution approach detailed above.
Step 1: Sorting the Array
First, we sort the array, which becomes [3, 3, 4, 5]
.
Step 2: Initializing Two Pointers
We set our two pointers i
and j
. The pointer i
starts at index 0
and j
starts at index 3
, the last index in the sorted array.
Step 3: Iterating and Pairing Persons to Boats
-
We compare the weights at
i
andj
, which arepeople[i] = 3
andpeople[j] = 5
. Since their sum is8
which is greater thanlimit
, they cannot go together. We need a boat forpeople[j]
, so we decrementj
to2
. -
Our pointers are now at
people[i] = 3
andpeople[j] = 4
. Their sum is7
, which again exceeds thelimit
. So, we placepeople[j]
in a new boat and decrementj
to1
. -
Now
people[i] = 3
andpeople[j] = 3
. Their sum is6
, still exceeding the limit. The third boat is used forpeople[j]
, and we decrementj
to0
. -
Finally, the pointers
i
andj
both point to the first element, which is3
. We have no choice but to use a fourth boat forpeople[i]
.
Step 4: Counting Boats
We needed one boat for each step, so the total number of boats used is 4
.
Step 5: Loop Termination
The loop terminates when i
is greater than j
. This happens after the fourth iteration in our example.
Using the approach, we determined that the minimum number of boats required to carry everybody in the array [3, 5, 3, 4]
with a limit of 5
is 4
. This example demonstrates the efficiency of the two-pointer technique in minimizing the number of boats used by attempting to pair the lightest and heaviest people on the same boat without surpassing the weight limit.
Solution Implementation
1from typing import List
2
3class Solution:
4 def num_rescue_boats(self, people: List[int], limit: int) -> int:
5 # Sort the list of people to organize by weight for optimal pairing.
6 people.sort()
7 # Initialize the count of rescue boats needed.
8 num_boats = 0
9 # Two pointers to keep track of the lightest and heaviest person not yet on a boat.
10 lightest_index, heaviest_index = 0, len(people) - 1
11
12 # Continue until all people have been assigned boats.
13 while lightest_index <= heaviest_index:
14 # If the lightest and heaviest person can share a boat, increase the lightest pointer.
15 if people[lightest_index] + people[heaviest_index] <= limit:
16 lightest_index += 1
17 # Always decrease the heaviest pointer since the heaviest person gets on a boat.
18 heaviest_index -= 1
19 # Increment the boat count as either one or two people have been assigned to a boat.
20 num_boats += 1
21
22 # Return the total number of boats needed.
23 return num_boats
24
25# Example usage:
26# solution = Solution()
27# print(solution.num_rescue_boats([3, 2, 2, 1], 3)) # Output: 3
28
1import java.util.Arrays; // Import Arrays class to use the sort method
2
3class Solution {
4
5 /**
6 * Returns the minimum number of boats required to rescue people.
7 *
8 * @param people An array representing the weight of each person.
9 * @param limit Maximum weight capacity a boat can carry.
10 * @return The minimum number of boats required.
11 */
12 public int numRescueBoats(int[] people, int limit) {
13 // Sort the array of people to organize weights in ascending order
14 Arrays.sort(people);
15
16 // Initialize the count of boats to 0
17 int boatCount = 0;
18
19 // Use two-pointer technique to pair the lightest and heaviest people
20 for (int lighter = 0, heavier = people.length - 1; lighter <= heavier; --heavier) {
21 // If the lightest and heaviest persons can share a boat, increment lighter pointer
22 if (people[lighter] + people[heavier] <= limit) {
23 lighter++;
24 }
25 // A boat is used, increment boat count
26 boatCount++;
27 }
28
29 // Return the total number of boats required after going through all the people
30 return boatCount;
31 }
32}
33
1#include <vector>
2#include <algorithm>
3
4class Solution {
5public:
6 // Method to find out the number of boats necessary to rescue all people.
7 // "people" is the list of weights of the people, and "limit" is the weight limit of the boat.
8 int numRescueBoats(vector<int>& people, int limit) {
9 // First, sort the list of people by their weights.
10 sort(people.begin(), people.end());
11
12 // Initialize the counter for the number of boats needed.
13 int numBoats = 0;
14
15 // Use two pointers, one at the beginning (lightest person) and one at the end (heaviest person).
16 int i = 0;
17 int j = people.size() - 1;
18
19 // Iterate until all people have been considered.
20 while (i <= j) {
21 // If the lightest and the heaviest person can share a boat, increment the pointer to the next lightest person.
22 if (people[i] + people[j] <= limit) {
23 i++;
24 }
25 // A boat is used for the heaviest person, decrement the pointer to the next heaviest person.
26 j--;
27
28 // Increment the counter for boats as either one or two people have used a boat.
29 numBoats++;
30 }
31
32 // Return the total number of boats needed.
33 return numBoats;
34 }
35};
36
1// Function to find out the number of boats necessary to rescue all people.
2// `people` is the list of weights of the people, and `limit` is the weight limit of the boat.
3function numRescueBoats(people: Array<number>, limit: number): number {
4 // First, sort the list of people by their weights in non-decreasing order.
5 people.sort((a, b) => a - b);
6
7 // Initialize the counter for the number of boats needed.
8 let numBoats: number = 0;
9
10 // Use two pointers, one at the beginning (lightest person) and one at the end (heaviest person).
11 let i: number = 0;
12 let j: number = people.length - 1;
13
14 // Iterate until all people have been considered.
15 while (i <= j) {
16 // If the lightest and the heaviest person can share a boat, increment the pointer to the next lightest person.
17 if (people[i] + people[j] <= limit) {
18 i++;
19 }
20 // A boat is used for the heaviest person, decrement the pointer to the next heaviest person.
21 j--;
22
23 // Increment the counter for boats as either one or two people have used a boat.
24 numBoats++;
25 }
26
27 // Return the total number of boats needed.
28 return numBoats;
29}
30
Time and Space Complexity
The given Python code implements an efficient algorithm to find the minimum number of boats required to rescue people based on their weights and the limit of a boat's capacity.
Time Complexity
The time complexity of the code is determined by the sorting operation and the while loop that is used to pair people onto boats.
- The
people.sort()
operation has a time complexity ofO(n log n)
, wheren
is the length of thepeople
list. Sorting is typically achieved using algorithms like Timsort in Python, which has this complexity. - The while loop runs in
O(n)
time since in the worst case, it could iterate over all elements once (wheren
is the number of people). Each operation inside the loop (checking condition and incrementing/decrementing counters) isO(1)
.
Combining both operations, the overall time complexity of the code is O(n log n) + O(n)
, which simplifies to O(n log n)
as the O(n log n)
term dominates O(n)
when n
is large.
Space Complexity
The space complexity of the algorithm is mainly due to the additional space required for sorting the people
list.
- The
people.sort()
operation is typically done in-place in Python, meaning that it doesn't require additional space proportional to its input, so it'sO(1)
space complexity. - The variables
i
,j
, andans
use a constant amount of extra spaceO(1)
.
Thus, the overall space complexity of the code is O(1)
, indicating that it uses a constant amount of space regardless of the input size.
Learn more about how to find time and space complexity quickly using problem constraints.
Consider the classic dynamic programming of longest increasing subsequence:
Find the length of the longest subsequence of a given sequence such that all elements of the subsequence are sorted in increasing order.
For example, the length of LIS for [50, 3, 10, 7, 40, 80]
is 4
and LIS is
[3, 7, 40, 80]
.
What is the recurrence relation?
Recommended Readings
Greedy Introduction div class responsive iframe iframe src https www youtube com embed WTslqPbj7I title YouTube video player frameborder 0 allow accelerometer autoplay clipboard write encrypted media gyroscope picture in picture web share allowfullscreen iframe div When do we use greedy Greedy algorithms tend to solve optimization problems Typically they will ask you to calculate the max min of some value Commonly you may see this phrased in the problem as max min longest shortest largest smallest etc These keywords can be identified by just scanning
Tech Interview Pattern Two Pointers Introduction If you prefer videos here's a super quick introduction to Two Pointers div class responsive iframe iframe src https www youtube com embed xZ4AfXHQ1VQ title YouTube video player frameborder 0 allow accelerometer autoplay clipboard write encrypted media gyroscope picture in picture allowfullscreen iframe div Two pointers is a common interview
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
Got a question?ย Ask the Monster Assistantย anything you don't understand.
Still not clear? ย Submitย the part you don't understand to our editors. Or join ourย Discord and ask the community.