LeetCode 365: Water and Jug Problem Solution in Python – A Step-by-Step Guide

Imagine you’re standing by a river with two jugs—one holds 5 liters, the other 3—and you need to figure out if you can measure exactly 4 liters using only fill, empty, and pour operations. That’s the puzzle of LeetCode 365: Water and Jug Problem, a medium-level problem that blends math and logic. Using Python, we’ll explore two solutions: the Best Solution—a mathematical approach with Bézout’s identity and GCD for O(log(min(x,y))) efficiency—and an Alternative Solution—a BFS method at O(x*y). With examples, clear code breakdowns, and a friendly vibe, this guide will help you measure that water. Let’s pour it out!

What Is LeetCode 365: Water and Jug Problem?

Section link icon

LeetCode 365: Water and Jug Problem gives you two jugs with capacities x and y, and a target amount z. You can fill, empty, or pour water between them, and the goal is to determine if it’s possible to measure exactly z liters using these operations. It’s a classic problem rooted in number theory and state exploration!

Problem Statement

  • Input:
    • jug1Capacity (x): Integer capacity of the first jug.
    • jug2Capacity (y): Integer capacity of the second jug.
    • targetCapacity (z): Integer target amount to measure.
  • Output: Boolean - True if z can be measured, False otherwise.
  • Rules:
    • Operations: Fill a jug, empty a jug, pour water from one jug to another.
    • Measure z using the total water in either or both jugs.

Constraints

  • 1 <= jug1Capacity, jug2Capacity <= 10⁶
  • 0 <= targetCapacity <= 10⁶

Examples

  • Input: jug1Capacity = 3, jug2Capacity = 5, targetCapacity = 4
    • Output: True
      • Fill 5L jug → (0,5), pour to 3L jug → (3,2), empty 3L → (0,2), pour 5L to 3L → (2,0), fill 5L → (2,5), pour to 3L → (3,4). Total = 4.
  • Input: jug1Capacity = 2, jug2Capacity = 6, targetCapacity = 5
    • Output: False
      • Possible sums are multiples of 2 (GCD), 5 is odd, impossible.
  • Input: jug1Capacity = 1, jug2Capacity = 2, targetCapacity = 3
    • Output: True
      • Fill 1L → (1,0), pour to 2L → (0,1), fill 1L → (1,1), pour to 2L → (0,2), fill 1L → (1,2). Total = 3.

These show the interplay of jug sizes—let’s solve it!

Understanding the Problem: Measuring Water

Section link icon

To tackle LeetCode 365 in Python, we need to: 1. Determine if z can be achieved using fill, empty, and pour operations. 2. Account for jug capacities x and y. 3. Ensure z is measurable as a total in one or both jugs.

A naive approach might simulate all operations, but we can use math or search. We’ll explore:

  • Best Solution: Bézout’s identity with GCD—O(log(min(x,y))) time, O(1) space—fast and elegant.
  • Alternative Solution: BFS—O(x*y) time, O(x*y) space—intuitive but slower.

Let’s measure with the best solution!

Best Solution: Bézout’s Identity with GCD

Section link icon

Why This Is the Best Solution

The Bézout’s identity approach runs in O(log(min(x,y))) time by using number theory: any amount measurable is a multiple of the greatest common divisor (GCD) of x and y, and z must be ≤ x + y and satisfy Bézout’s equation (ax + by = z). It’s the most efficient—avoiding exhaustive state exploration!

How It Works

Think of it like a math trick:

  • Bézout’s Identity: For integers a and b, if z = ax + by, then z must be a multiple of gcd(x,y).
  • Conditions:
    • z ≤ x + y (total capacity).
    • z % gcd(x,y) == 0 (must be achievable).
  • GCD: Compute using Euclidean algorithm.
  • Result: True if conditions hold, False otherwise.

It’s like finding the rhythm of pourable amounts!

Step-by-Step Example

For x=3, y=5, z=4:

  • GCD: gcd(3,5) = 1 (3,5 are coprime).
  • Check:
    • z=4 ≤ 3+5=8 ✓.
    • 4 % 1 = 0 ✓.
  • Bézout: 4 = 3*(-2) + 5*2 (possible with negative a, b allowed in theory).
  • Result: True (operations exist: e.g., fill 5, pour to 3, empty 3, pour 2 to 3, fill 5, pour to 3 → (3,4)).

For x=2, y=6, z=5:

  • GCD: gcd(2,6) = 2.
  • Check:
    • 5 ≤ 2+6=8 ✓.
    • 5 % 2 = 1 ≠ 0 ✗.
  • Result: False (only multiples of 2 possible).

Code with Explanation

Here’s the Python code using GCD:

class Solution:
    def canMeasureWater(self, jug1Capacity: int, jug2Capacity: int, targetCapacity: int) -> bool:
        # Helper to compute GCD using Euclidean algorithm
        def gcd(a, b):
            while b:
                a, b = b, a % b
            return a

        x, y, z = jug1Capacity, jug2Capacity, targetCapacity

        # Check basic conditions
        if z > x + y:  # Target exceeds total capacity
            return False
        if z == 0:  # Empty is always possible
            return True
        if x == 0 or y == 0:  # One jug empty
            return z == 0 or z == max(x, y)

        # Check if z is a multiple of gcd(x, y)
        return z % gcd(x, y) == 0

Explanation

  • Lines 3-7: gcd: Euclidean algorithm—e.g., gcd(3,5): 5%3=2, 3%2=1, 2%1=0 → 1.
  • Line 10: Rename for clarity: x=jug1, y=jug2, z=target.
  • Lines 12-16: Base cases:
    • z > x + y: Impossible if target exceeds total.
    • z = 0: True (empty state).
    • x or y = 0: True only if z = 0 or z = other jug’s capacity.
  • Line 18: Core check: z must be a multiple of gcd(x,y) per Bézout’s identity.
  • Time Complexity: O(log(min(x,y)))—GCD computation.
  • Space Complexity: O(1)—no extra space.

This is like a mathematical water key—unlock the solution fast!

Alternative Solution: BFS

Section link icon

Why an Alternative Approach?

The BFS (Breadth-First Search) solution runs in O(x*y) time by exploring all possible states of the jugs. It’s slower but explicitly simulates operations—great for understanding the process or verifying the math!

How It Works

  • States: Track (jug1, jug2) water amounts.
  • BFS: Use a queue to explore states via fill, empty, pour.
  • Check: If total water = z, return True.

Step-by-Step Example

For x=3, y=5, z=4:

  • Init: Queue = [(0,0)], Seen = {(0,0)}.
  • (0,0):
    • Fill 3L: (3,0) → Queue, Seen.
    • Fill 5L: (0,5) → Queue, Seen.
  • (3,0):
    • Pour 3L→5L: (0,3).
    • Total=3 < 4.
  • (0,5):
    • Pour 5L→3L: (3,2).
    • Total=5 > 4.
  • (3,2):
    • Total=5 > 4.
    • Empty 3L: (0,2).
  • (0,2):
    • Fill 3L: (3,2).
    • Total=2 < 4.
  • (3,2):
    • Pour 3L→5L: (0,5) (seen).
    • Pour 5L→3L: (3,4).
  • (3,4):
    • Total=7 > 4.
    • Empty 3L: (0,4).
  • (0,4):
    • Total=4 ✓, return True.

Code with Explanation

from collections import deque

class Solution:
    def canMeasureWater(self, jug1Capacity: int, jug2Capacity: int, targetCapacity: int) -> bool:
        x, y, z = jug1Capacity, jug2Capacity, targetCapacity

        if z > x + y:
            return False
        if z == 0:
            return True

        # BFS with state (jug1, jug2)
        queue = deque([(0, 0)])
        seen = {(0, 0)}

        while queue:
            j1, j2 = queue.popleft()
            total = j1 + j2

            if total == z:
                return True

            # Possible operations
            states = [
                (x, j2),  # Fill jug1
                (j1, y),  # Fill jug2
                (0, j2),  # Empty jug1
                (j1, 0),  # Empty jug2
                # Pour jug1 to jug2
                (j1 - min(j1, y - j2), j2 + min(j1, y - j2)),
                # Pour jug2 to jug1
                (j1 + min(j2, x - j1), j2 - min(j2, x - j1))
            ]

            for next_j1, next_j2 in states:
                if 0 <= next_j1 <= x and 0 <= next_j2 <= y and (next_j1, next_j2) not in seen:
                    queue.append((next_j1, next_j2))
                    seen.add((next_j1, next_j2))

        return False

Explanation

  • Lines 7-11: Base cases: z > total capacity or z = 0.
  • Lines 13-14: Start BFS with (0,0) state.
  • Lines 16-34: BFS loop:
    • Pop current state, check if total = z.
    • Generate next states: fill, empty, pour (calculate water transfer).
    • Add valid unseen states to queue.
  • Time: O(x*y)—number of possible states.
  • Space: O(x*y)—queue and seen set.

It’s like exploring all water paths!

Comparing the Two Solutions

Section link icon
  • Bézout’s Identity with GCD (Best):
    • Time: O(log(min(x,y)))—GCD computation.
    • Space: O(1)—minimal.
    • Pros: Super fast, elegant.
    • Cons: Requires math insight.
  • BFS (Alternative):
    • Time: O(x*y)—state exploration.
    • Space: O(x*y)—queue and set.
    • Pros: Explicit, verifiable.
    • Cons: Slow for large capacities.

GCD wins for efficiency!

Additional Examples and Edge Cases

Section link icon
  • x=1, y=1, z=1: True.
  • x=0, y=5, z=5: True.
  • x=2, y=4, z=3: False.

Both handle these, GCD is faster.

Complexity Breakdown

Section link icon
  • GCD:
    • Time: O(log(min(x,y))).
    • Space: O(1).
  • BFS:
    • Time: O(x*y).
    • Space: O(x*y).

GCD excels for speed.

Key Takeaways

Section link icon
  • GCD: Math solves it—fast and smart!
  • BFS: Explore all paths—clear and thorough!
  • Jugs: Logic meets practice.
  • Python Tip: Math and queues rock—see [Python Basics](/python/basics).

Final Thoughts: Measure That Water

Section link icon

LeetCode 365: Water and Jug Problem in Python is a jug-juggling challenge. GCD with Bézout’s identity is the efficiency champ, while BFS builds intuition. Want more? Try LeetCode 279: Perfect Squares or LeetCode 373: Find K Pairs with Smallest Sums. Ready to pour? Visit Solve LeetCode 365 on LeetCode and measure that water today!