1833. Maximum Ice Cream Bars
Problem Description
On a hot summer day, a boy is looking to buy some ice cream bars from a store. The store has n
different types of ice cream bars, each with its own price. The prices are given in an array costs
with n
elements, where costs[i]
represents the price of the i-th
ice cream bar in coins. The boy has a limited amount of money, specifically coins
coins, which he will use to purchase as many ice cream bars as he can.
The key point here is that the boy is allowed to buy the ice cream bars in any order he prefers. The main challenge is to determine the maximum number of ice cream bars the boy can buy without exceeding the number of coins he has.
To solve the problem, we need to formulate an approach that ensures we can get the maximum number of bars under the given constraints.
Intuition
Considering that we aim to maximize the number of ice cream bars the boy can buy, it makes sense to start by purchasing the cheapest available options. This strategy ensures that we get the most out of the coins available to us.
We follow a Greedy approach to solve this problem. The Greedy algorithm is a straightforward, intuitive method where we always choose the next most favorable option, hoping that this will lead to an optimal overall solution. In this case, the most favorable option is to buy the cheapest ice cream bar first.
To implement this Greedy approach efficiently, we sort the costs
array in ascending order. Then, we iterate through this sorted list and keep subtracting the cost of each ice cream bar from the total coins we have until we can't afford to buy the next bar.
At each step, we're making the optimal local decision to buy the cheapest available ice cream bar, and as a result, we end up with the maximum number of bars that can be bought with the coins available.
Solution Approach
The solution follows a Greedy algorithm, which is perfect for this scenario since we want to buy as many items as possible given a constraint (the coins).
Here's a breakdown of how the algorithm is implemented, referring to the given solution approach:
-
Sort the
costs
Array: The first step is to sort the array of ice cream costs. This is essential because it allows us to start with the least expensive options, ensuring we can buy as many as possible.costs.sort()
This operation uses an
O(n log n)
sorting algorithm wheren
is the number of ice cream bars (n
being the length ofcosts
). -
Iterate through the Sorted
costs
: Once we have thecosts
array sorted, we iterate through it to simulate purchasing the ice cream bars from the cheapest to the most expensive.for i, c in enumerate(costs):
We use a
for
loop combined withenumerate
to go through each price (c
) along with its index (i
) in the sortedcosts
array. -
Check if the Next Ice Cream Can Be Bought: For each price in the array, we check if we have enough coins to buy the ice cream at the current price. If we do not have enough, we simply return the number of ice cream bars bought so far, which would correspond to the index
i
.if coins < c: return i
If the ice cream bar can be purchased, we subtract its cost from our total coins.
coins -= c
We repeat this step until we run out of coins or have bought all the ice cream bars.
-
Return the Total Bars Bought: If we break out of the loop because we've run out of coins, we will have already returned the number of ice cream bars bought. If we complete the loop without running out of coins, it means we could buy all available ice cream bars. Therefore, we return the total number of ice cream bars (
len(costs)
).return len(costs)
The use of sorting, followed by a simple linear scan of the array, allows this solution to be effective and to the point. This algorithm's efficiency relies heavily on the sorting step, which is why sorting algorithms are such a critical part of most Greedy approaches to problem-solving.
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 consider a small example to illustrate the solution approach. Suppose we have an array of ice cream costs costs = [1, 3, 2, 4, 1]
and the boy has coins = 7
.
Here's how we apply the Greedy algorithm to maximize the number of ice cream bars the boy can buy:
-
Sort the
costs
Array: First, we sort thecosts
array. After sorting, we get[1, 1, 2, 3, 4]
. This sorting will allow us to consider the cheapest ice cream bars first. -
Iterate through the Sorted
costs
: We start iterating from the first element of the sorted array.For
i = 0
(first ice cream bar),c = 1
. We check if the boy can afford to buy this ice cream bar. Sincecoins = 7
andc = 1
, he can afford it. Thus, we subtract the cost of this ice cream bar from his coins:coins = 7 - 1 = 6
.For
i = 1
(second ice cream bar),c = 1
. He can still afford it, so again we subtract the cost:coins = 6 - 1 = 5
.For
i = 2
(third ice cream bar),c = 2
. He can afford it, so we subtract the cost:coins = 5 - 2 = 3
.For
i = 3
(fourth ice cream bar),c = 3
. He can afford it, so we subtract the cost:coins = 3 - 3 = 0
.For
i = 4
(fifth ice cream bar),c = 4
. Now the boy hascoins = 0
and cannot afford to buy this ice cream bar. -
Check if Next Ice Cream Can Be Bought & Return Total Bars Bought: As soon as the boy can no longer afford an ice cream bar, we return the number he has bought so far. In this case, after buying the first four ice cream bars, he's left with zero coins and is unable to buy the fifth one that costs 4 coins.
Therefore, the maximum number of ice cream bars the boy can buy is 4
.
Solution Implementation
1class Solution:
2 def maxIceCream(self, ice_cream_costs: List[int], total_coins: int) -> int:
3 # Sort the ice cream costs in non-decreasing order
4 ice_cream_costs.sort()
5
6 # Initialize a variable to count the number of ice creams bought
7 ice_creams_bought = 0
8
9 # Iterate over the sorted ice cream costs
10 for cost in ice_cream_costs:
11 # If the current ice cream cost is more than the total coins,
12 # we cannot buy this ice cream, so we return the count.
13 if total_coins < cost:
14 return ice_creams_bought
15 # Otherwise, we can buy this ice cream
16 # so we subtract its cost from the total coins
17 total_coins -= cost
18 # and increment the count of ice creams bought
19 ice_creams_bought += 1
20
21 # If we were able to buy all the ice creams without running out of coins,
22 # return the total number of ice creams bought,
23 # which is the length of the costs list
24 return ice_creams_bought
25
1import java.util.Arrays; // Import Arrays utility for sorting
2
3class Solution {
4
5 // Method for calculating maximum number of ice creams that can be bought with the given coins
6 public int maxIceCream(int[] costs, int coins) {
7 // Sort the array to purchase cheaper ice creams first
8 Arrays.sort(costs);
9
10 // Get the number of ice cream prices in the array
11 int numOfIceCreams = costs.length;
12
13 // Initialize the count of ice creams bought
14 int count = 0;
15
16 // Iterate over the sorted array
17 for (int i = 0; i < numOfIceCreams; ++i) {
18 // If the current ice cream cost is more than the available coins
19 // Return the count of ice creams bought so far
20 if (coins < costs[i]) {
21 return count;
22 }
23
24 // Subtract the cost of the current ice cream from the available coins
25 coins -= costs[i];
26
27 // Increment the count of ice creams bought
28 count++;
29 }
30
31 // Return the total count of ice creams that can be bought
32 // This line is reached when all ice creams can be bought with the available coins
33 return count;
34 }
35}
36
1#include <vector>
2#include <algorithm>
3
4class Solution {
5public:
6 // Function to calculate the maximum number of ice creams that can be bought with `coins`.
7 int maxIceCream(vector<int>& costs, int coins) {
8 // Sort the `costs` vector in non-decreasing order.
9 std::sort(costs.begin(), costs.end());
10
11 // Initialize the counter for the number of ice creams.
12 int numIceCreams = 0;
13
14 // Iterate through the sorted ice cream costs.
15 for (int cost : costs) {
16 // If we don't have enough coins to buy the current ice cream, break the loop.
17 if (coins < cost) break;
18
19 // Subtract the cost from our coins and increment the count of ice creams.
20 coins -= cost;
21 numIceCreams++;
22 }
23
24 // Return the number of ice creams purchased.
25 return numIceCreams;
26 }
27};
28
1/**
2 * Function to calculate the maximum number of ice creams that can be bought
3 * with a given amount of coins.
4 * @param {number[]} iceCreamCosts - The array of costs of different ice creams.
5 * @param {number} totalCoins - The total number of coins you have.
6 * @returns {number} The maximum number of ice creams that can be bought.
7 */
8function maxIceCream(iceCreamCosts: number[], totalCoins: number): number {
9 // Sort the ice cream costs in ascending order
10 iceCreamCosts.sort((a, b) => a - b);
11
12 // The number of different ice creams available
13 const numberOfIceCreams = iceCreamCosts.length;
14
15 // Loop through each ice cream cost
16 for (let i = 0; i < numberOfIceCreams; ++i) {
17 // If the current ice cream cost is more than the remaining coins, return the count of ice creams bought so far
18 if (totalCoins < iceCreamCosts[i]) {
19 return i;
20 }
21 // Subtract the cost of the current ice cream from the total coins
22 totalCoins -= iceCreamCosts[i];
23 }
24
25 // If all ice creams could be bought with the available coins, return the total number of ice creams
26 return numberOfIceCreams;
27}
28
Time and Space Complexity
The time complexity of the provided code is O(n log n)
. This complexity arises from the fact that the main operation in this algorithm is sorting the costs
array, which is typically implemented with an algorithm like quicksort or mergesort that has an O(n log n)
complexity in Python’s sort function.
After sorting, the code iterates through the sorted costs
array only once, with each iteration being a constant time operation. However, since this linear scan (O(n)
) is dominated by the sorting step, it does not change the overall time complexity of the algorithm.
The space complexity of the code is O(log n)
. This is because the space complexity is attributed to the space that is used during the sorting of the array. Most sorting algorithms like quicksort and mergesort typically require O(log n)
space on the call stack for recursive function calls.
Learn more about how to find time and space complexity quickly using problem constraints.
Which of the following is a good use case for backtracking?
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
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
LeetCode Patterns Your Personal Dijkstra's Algorithm to Landing Your Dream Job The goal of AlgoMonster is to help you get a job in the shortest amount of time possible in a data driven way We compiled datasets of tech interview problems and broke them down by patterns This way we
Want a Structured Path to Master System Design Too? Don’t Miss This!