LeetCode 135: Candy Solution in Python Explained

Distributing candies to children based on their ratings might feel like balancing fairness in a classroom, and LeetCode 135: Candy is a medium-level challenge that makes it intriguing! Given an integer array ratings representing children’s ratings, you need to return the minimum number of candies to distribute, ensuring each child gets at least one candy, and children with higher ratings than their neighbors get more candies. In this blog, we’ll solve it with Python, exploring two solutions—Two-Pass Greedy (our best solution) and Single-Pass with Peaks and Valleys (a clever alternative). With step-by-step examples, detailed code breakdowns, and tips, you’ll master this problem. Let’s share those candies!

Problem Statement

Section link icon

In LeetCode 135, you’re given an integer array ratings of length n, where ratings[i] is the rating of the i-th child in a line. Your task is to return the minimum number of candies to distribute, following these rules: 1. Each child gets at least 1 candy. 2. If a child’s rating is higher than their neighbor’s, they must get more candies than that neighbor.

This differs from circular traversal like LeetCode 134: Gas Station, focusing on a linear array with local comparisons rather than a circuit.

Constraints

  • 1 <= ratings.length <= 2 * 10^4
  • 0 <= ratings[i] <= 2 * 10^4

Example

Let’s see some cases:

Input: ratings = [1,0,2]
Output: 5
Explanation: Candies: [2,1,2]
<ul>
<li>Child 0 (1) > Child 1 (0), gets 2.</li>
<li>Child 1 (0), minimum 1.</li>
<li>Child 2 (2) > Child 1 (0), gets 2.</li>
<li>Total: 2+1+2 = 5.</li>
</ul>
Input: ratings = [1,2,2]
Output: 4
Explanation: Candies: [1,2,1]
<ul>
<li>Child 0 (1) < Child 1 (2), gets 1.</li>
<li>Child 1 (2), gets 2 (more than 0).</li>
<li>Child 2 (2) = Child 1 (2), gets 1 (minimum).</li>
<li>Total: 1+2+1 = 4.</li>
</ul>
Input: ratings = [1,3,2,1]
Output: 7
Explanation: Candies: [1,3,2,1]
<ul>
<li>Child 0 (1) < Child 1 (3), gets 1.</li>
<li>Child 1 (3) > Child 0 (1) and Child 2 (2), gets 3.</li>
<li>Child 2 (2) > Child 3 (1), gets 2.</li>
<li>Child 3 (1), gets 1.</li>
<li>Total: 1+3+2+1 = 7.</li>
</ul>

These examples show we’re minimizing candies while satisfying neighbor rules.

Understanding the Problem

Section link icon

How do you solve LeetCode 135: Candy in Python? Take ratings = [1,0,2]. We need the minimum candies: child 0 (rating 1) > child 1 (rating 0), child 2 (rating 2) > child 1 (rating 0), so [2,1,2] gives 5 candies. For [1,2,2], equal ratings allow minimum candies after a peak, so [1,2,1] gives 4. This isn’t a path sum problem like LeetCode 113: Path Sum II; it’s about array-based greedy allocation. We’ll use: 1. Two-Pass Greedy: O(n) time, O(n) space—our best solution. 2. Single-Pass with Peaks and Valleys: O(n) time, O(1) space—an alternative.

Let’s dive into the best solution.

Best Solution: Two-Pass Greedy Approach

Section link icon

Explanation

Two-Pass Greedy assigns candies in two sweeps: a left-to-right pass ensures each child gets more than their left neighbor if their rating is higher, and a right-to-left pass adjusts for right neighbors, taking the maximum from both passes. This is the best solution due to its O(n) time complexity, clear logic ensuring all conditions are met, and straightforward implementation, making it reliable and easy to understand.

For [1,0,2]:

  • Left pass: [1,1,2] (2 > 0).
  • Right pass: [2,1,2] (1 > 0, 2 > 1 adjusted left).
  • Total: 5.

Step-by-Step Example

Example 1: ratings = [1,0,2]

Goal: Return 5.

  • Start: candies = [1,1,1], n = 3.
  • Left Pass:
    • i=1: 0 < 1, candies[1] = 1.
    • i=2: 2 > 0, candies[2] = 1+1 = 2.
    • candies = [1,1,2].
  • Right Pass:
    • i=1: 0 < 2, candies[1] = 1.
    • i=0: 1 > 0, candies[0] = max(1, 1+1) = 2.
    • candies = [2,1,2].
  • Sum: 2+1+2 = 5.
  • Finish: Return 5.

Example 2: ratings = [1,2,2]

Goal: Return 4.

  • Start: candies = [1,1,1].
  • Left Pass:
    • i=1: 2 > 1, candies[1] = 2.
    • i=2: 2 = 2, candies[2] = 1.
    • candies = [1,2,1].
  • Right Pass:
    • i=1: 2 = 2, candies[1] = 2.
    • i=0: 1 < 2, candies[0] = 1.
    • candies = [1,2,1].
  • Sum: 1+2+1 = 4.
  • Finish: Return 4.

Example 3: ratings = [1,3,2,1]

Goal: Return 7.

  • Start: candies = [1,1,1,1].
  • Left Pass:
    • i=1: 3 > 1, candies[1] = 2.
    • i=2: 2 < 3, candies[2] = 1.
    • i=3: 1 < 2, candies[3] = 1.
    • candies = [1,2,1,1].
  • Right Pass:
    • i=2: 2 > 1, candies[2] = max(1, 1+1) = 2.
    • i=1: 3 > 2, candies[1] = max(2, 2+1) = 3.
    • i=0: 1 < 3, candies[0] = 1.
    • candies = [1,3,2,1].
  • Sum: 1+3+2+1 = 7.
  • Finish: Return 7.

How the Code Works (Two-Pass Greedy) – Detailed Line-by-Line Explanation

Here’s the Python code with a thorough breakdown:

def candy(ratings: list[int]) -> int:
    # Line 1: Get number of children
    n = len(ratings)
    # Length of ratings array (e.g., 3 for [1,0,2])

    # Line 2: Initialize candies array with minimum 1
    candies = [1] * n
    # Each child starts with 1 candy (e.g., [1,1,1])

    # Line 3: Left-to-right pass
    for i in range(1, n):
        # i = 1 to n-1 (e.g., 1 to 2)

        # Line 4: Check if current rating > previous
        if ratings[i] > ratings[i-1]:
            # If higher (e.g., 2 > 0), give more than left neighbor
            candies[i] = candies[i-1] + 1
            # e.g., candies[2] = 1+1 = 2

    # Line 5: Right-to-left pass
    for i in range(n-2, -1, -1):
        # i = n-2 to 0 (e.g., 1 to 0)

        # Line 6: Check if current rating > next
        if ratings[i] > ratings[i+1]:
            # If higher (e.g., 1 > 0), ensure more than right neighbor
            candies[i] = max(candies[i], candies[i+1] + 1)
            # e.g., candies[0] = max(1, 1+1) = 2
            # Max ensures left pass isn’t overridden unless needed

    # Line 7: Return total candies
    return sum(candies)
    # Sums array (e.g., [2,1,2] → 5)

This detailed breakdown clarifies how the two-pass greedy approach efficiently assigns candies.

Alternative: Single-Pass with Peaks and Valleys Approach

Section link icon

Explanation

Single-Pass with Peaks and Valleys scans the array once, counting candies by identifying peaks (local maxima) and valleys (local minima), calculating contributions based on increasing/decreasing sequences. It’s a clever alternative, achieving O(n) time with O(1) space, but it’s more complex to implement and understand.

For [1,0,2]:

  • Up: 1 (1->0), Down: 1+1 (0->2), Peak: 2-1=1, Total: 1+2=3+2=5.

Step-by-Step Example (Alternative)

For [1,0,2]:

  • Start: candies = 0, prev = 1.
  • Step 1: i=1, 0 < 1, down = 1, candies = 1.
  • Step 2: i=2, 2 > 0, up = 1, peak = max(1,1)-1=0, candies = 1+1+2=4+1=5.
  • Finish: Return 5.

How the Code Works (Peaks and Valleys)

def candyPeaksValleys(ratings: list[int]) -> int:
    if not ratings:
        return 0
    n = len(ratings)
    if n == 1:
        return 1
    candies = up = down = 0
    for i in range(1, n):
        if ratings[i] > ratings[i-1]:
            up += 1
            down = 0
            candies += up + max(up, down) - (1 if up > down else 0)
        elif ratings[i] < ratings[i-1]:
            down += 1
            up = 0
            candies += down + max(up, down) - (1 if up > down else 0)
        else:
            up = down = 0
            candies += 1
    return candies + 1  # Add base candy for first child

Complexity

  • Two-Pass Greedy:
    • Time: O(n) – two linear passes.
    • Space: O(n) – candies array.
  • Single-Pass with Peaks and Valleys:
    • Time: O(n) – single pass.
    • Space: O(1) – constant space.

Efficiency Notes

Two-Pass Greedy is the best solution with O(n) time and O(n) space, offering clarity and reliability—Single-Pass with Peaks and Valleys matches time complexity with O(1) space, but its complexity makes it less intuitive, though more space-efficient.

Key Insights

  • Greedy: Local comparisons suffice.
  • Two-Pass: Ensures all conditions.
  • Peaks: Handles slopes efficiently.

Additional Example

Section link icon

For ratings = [1,2,3,2]:

  • Goal: 7.
  • Two-Pass: [1,2,3,1], total 7.

Edge Cases

Section link icon
  • Single Child: [1]1.
  • All Equal: [1,1,1]3.
  • Decreasing: [3,2,1]6.

Both solutions handle these well.

Final Thoughts

Section link icon

LeetCode 135: Candy in Python is a delightful greedy challenge. The Two-Pass Greedy solution excels with its efficiency and simplicity, while Single-Pass with Peaks and Valleys offers a space-optimized twist. Want more? Try LeetCode 134: Gas Station for another greedy problem or LeetCode 94: Binary Tree Inorder Traversal for tree basics. Ready to practice? Solve LeetCode 135 on LeetCode with [1,0,2], aiming for 5—test your skills now!