LeetCode 283: Move Zeroes Solution in Python – A Step-by-Step Guide

Imagine you’re tidying up a row of numbers, like [0, 1, 0, 3, 12], and your task is to sweep all the zeros to the end while keeping the other numbers in their original order—so it becomes [1, 3, 12, 0, 0]. That’s the essence of LeetCode 283: Move Zeroes, an easy-level problem that’s all about rearranging an array in-place with a clever twist. Using Python, we’ll explore two solutions: the Best Solution, a two-pointer approach that’s fast and efficient, and an Alternative Solution, a bubble-like shifting method that’s simpler but slower. With detailed examples, clear code, and a friendly tone—especially for the two-pointer breakdown—this guide will help you master this problem, whether you’re new to coding or brushing up your skills. Let’s start moving those zeros!

What Is LeetCode 283: Move Zeroes?

Section link icon

In LeetCode 283: Move Zeroes, you’re given an integer array nums, and your goal is to modify it in-place so that all zeros are pushed to the end, while the non-zero elements stay in their relative order. For example, [0, 1, 0, 3, 12] becomes [1, 3, 12, 0, 0]—no new array allowed, just shuffle the existing one. This problem tests your ability to manipulate arrays efficiently, sharing a spirit with classics like LeetCode 27: Remove Element, but with a focus on preserving order and relocating zeros.

Problem Statement

  • Input: An integer array nums.
  • Output: The array nums, modified in-place with all zeros at the end.
  • Rules: Move zeros to the end; keep non-zero elements’ relative order; no extra array.

Constraints

  • nums length: 1 to 10⁴ (10,000).
  • nums[i]: -2³¹ to 2³¹ - 1.

Examples

  • Input: nums = [0, 1, 0, 3, 12]
    • Output: [1, 3, 12, 0, 0]
  • Input: nums = [0]
    • Output: [0]
  • Input: nums = [1, 2, 3]
    • Output: [1, 2, 3]

Understanding the Problem: Sweeping the Zeros

Section link icon

To solve LeetCode 283: Move Zeroes in Python, we need to push all zeros to the end of nums while keeping non-zeros in their original sequence—all without a second array. For [0, 1, 0, 3, 12], we want [1, 3, 12] up front, then [0, 0] at the back. A naive idea might be to shift zeros one-by-one, but that’s slow. Instead, we’ll use: 1. Best Solution (Two-Pointer): O(n) time, O(1) space—quick and clean. 2. Alternative Solution (Bubble-Like Shifting): O(n²) time—simple but inefficient.

Let’s dive into the two-pointer solution with a beginner-friendly explanation.

Best Solution: Two-Pointer Approach

Section link icon

Why This Is the Best Solution

The two-pointer approach is the top pick for LeetCode 283 because it’s blazing fast—O(n) time—and uses O(1) space, making just one pass through the array (plus a quick zero-fill). It’s like using one pointer to gather non-zeros at the front and another to track where they go, then filling the rest with zeros. For beginners, it’s a neat trick that feels like magic once you see it in action!

How It Works

Picture this as sorting laundry: you’ve got a pile (the array), and you want all the “good clothes” (non-zeros) at the front of your basket, with the “empty hangers” (zeros) pushed to the back. You use one hand to find non-zeros and another to place them—then toss zeros in the leftover space. Here’s how it works, step-by-step:

  • Step 1: Gather Non-Zeros:
    • Use a pointer non_zero_pos to track where the next non-zero element should go (starts at 0).
    • Loop through the array: when you find a non-zero, place it at non_zero_pos and increment it.
  • Step 2: Fill Zeros:
    • After non-zeros are in front, fill the rest of the array (from non_zero_pos to the end) with zeros.
  • Step 3: Why It Works:
    • Non-zeros stay in order because we copy them as we find them.
    • Zeros naturally end up at the end since we only move non-zeros forward.
    • In-place means we’re just rearranging within nums.
  • Step 4: Result:
    • Array has all non-zeros first, zeros last—no extra space needed.

It’s like herding sheep to one side and letting the empty field fill the rest!

Step-by-Step Example

Example: nums = [0, 1, 0, 3, 12]

  • Initial: [0, 1, 0, 3, 12], non_zero_pos = 0.
  • Pass 1 (Gather Non-Zeros):
    • i=0: 0—skip.
    • i=1: 1—nums[0] = 1, non_zero_pos = 1.
    • i=2: 0—skip.
    • i=3: 3—nums[1] = 3, non_zero_pos = 2.
    • i=4: 12—nums[2] = 12, non_zero_pos = 3.
  • After Pass 1: [1, 3, 12, 3, 12] (last two spots unchanged yet).
  • Pass 2 (Fill Zeros):
    • From non_zero_pos = 3 to end: nums[3] = 0, nums[4] = 0.
  • Result: [1, 3, 12, 0, 0].

Example: nums = [0, 0, 1]

  • Pass 1: i=2: 1—nums[0] = 1, non_zero_pos = 1.
  • After: [1, 0, 1].
  • Pass 2: nums[1] = 0, nums[2] = 0.
  • Result: [1, 0, 0].

Code with Detailed Line-by-Line Explanation

Here’s the Python code, explained simply:

class Solution:
    def moveZeroes(self, nums: List[int]) -> None:
        # Step 1: Pointer for next non-zero position
        non_zero_pos = 0

        # Step 2: Move all non-zeros to the front
        for i in range(len(nums)):
            if nums[i] != 0:
                nums[non_zero_pos] = nums[i]
                non_zero_pos += 1

        # Step 3: Fill the rest with zeros
        for i in range(non_zero_pos, len(nums)):
            nums[i] = 0
  • Line 4: non_zero_pos starts at 0—where the first non-zero goes.
  • Line 6-9: Loop through nums:
    • If nums[i] isn’t zero, copy it to non_zero_pos and move the pointer up.
  • Line 11-12: From non_zero_pos to the end, set each spot to 0.
  • No return: Modifies nums in-place.
  • Time Complexity: O(n)—one pass for non-zeros, one for zeros.
  • Space Complexity: O(1)—just a pointer, no extra array.

This solution is like a swift cleanup—non-zeros first, zeros last!

Alternative Solution: Bubble-Like Shifting

Section link icon

Why an Alternative Approach?

The bubble-like shifting method moves zeros to the end by swapping them with the next non-zero element, like bubbling them through the array. It’s O(n²) time—slower—but super intuitive, making it a great way to grasp the problem before jumping to the optimized approach.

How It Works

Imagine pushing each zero to the right, one swap at a time, until it hits the end—like bubbles rising in water:

  • Step 1: Loop through the array.
  • Step 2: When you find a zero, swap it with the next element, repeating until it’s at the end or hits another zero.
  • Step 3: Continue until all zeros are at the end.

It’s a slow shuffle but gets the job done!

Step-by-Step Example

Example: nums = [0, 1, 0, 3, 12]

  • Initial: [0, 1, 0, 3, 12].
  • i=0: 0—swap with 1: [1, 0, 0, 3, 12].
  • i=1: 0—swap with 0, then 3: [1, 0, 3, 0, 12].
  • i=2: 3—skip.
  • i=3: 0—swap with 12: [1, 0, 3, 12, 0].
  • i=1: 0—swap with 3, 12, 0: [1, 3, 0, 12, 0], then [1, 3, 12, 0, 0].
  • Result: [1, 3, 12, 0, 0].

Code for Bubble-Like Shifting

class Solution:
    def moveZeroes(self, nums: List[int]) -> None:
        n = len(nums)
        for i in range(n):
            if nums[i] == 0:
                # Bubble the zero to the right
                for j in range(i, n - 1):
                    if nums[j + 1] != 0 or j == n - 2:
                        nums[j], nums[j + 1] = nums[j + 1], nums[j]
  • Time Complexity: O(n²)—nested loops for each zero.
  • Space Complexity: O(1)—in-place swaps.

It’s a steady march, but not the fastest!

Comparing the Two Solutions

Section link icon
  • Two-Pointer (Best):
    • Pros: O(n) time, O(1) space, one pass.
    • Cons: Slightly trickier to visualize.
  • Bubble-Like (Alternative):
    • Pros: Easy to understand, step-by-step.
    • Cons: O(n²) time, too slow for big arrays.

Two-pointer wins hands down.

Additional Examples and Edge Cases

Section link icon
  • All Zeros: [0, 0] → [0, 0].
  • No Zeros: [1, 2, 3] → [1, 2, 3].
  • Single Element: [0] → [0].

Both handle these perfectly.

Complexity Breakdown

Section link icon
  • Two-Pointer: Time O(n), Space O(1).
  • Bubble: Time O(n²), Space O(1).

Two-pointer is the speed king.

Key Takeaways

Section link icon
  • Two-Pointer: Gather and fill—fast!
  • Bubble: Swap and shift—slow but sure.
  • In-Place: No extra space needed.
  • Python Tip: Loops and swaps shine—see [Python Basics](/python/basics).

Final Thoughts: Zero In on Success

Section link icon

LeetCode 283: Move Zeroes in Python is a fun array challenge. The two-pointer solution zips through efficiently, while bubbling offers a gentle intro. Want more? Try LeetCode 27: Remove Element or LeetCode 88: Merge Sorted Array. Ready to sweep? Head to Solve LeetCode 283 on LeetCode and move those zeros today!