LeetCode 324: Wiggle Sort II Solution in Python – A Step-by-Step Guide

Imagine you’re arranging a row of numbers so they zigzag—each number alternately greater or less than its neighbors—like a playful dance of digits. That’s the challenge of LeetCode 324: Wiggle Sort II, a medium-level problem that blends sorting with array manipulation. Using Python, we’ll explore two solutions: the Best Solution, a clever O(n) time virtual indexing approach with partitioning, and an Alternative Solution, a straightforward sorting method that’s easier to grasp but slower. With detailed examples, clear code, and a friendly tone—especially for the virtual indexing breakdown—this guide will help you wiggle those numbers, whether you’re new to coding or leveling up. Let’s dive into the array and make it dance!

What Is LeetCode 324: Wiggle Sort II?

Section link icon

In LeetCode 324: Wiggle Sort II, you’re given an integer array nums, and your task is to reorder it in-place into a wiggle sequence: nums[0] < nums[1] > nums[2] < nums[3] > .... For example, [1,5,1,1,6,4] can become [1,6,1,5,1,4]. The problem guarantees a solution exists, but it must handle duplicates and ensure strict inequalities. This builds on concepts from LeetCode 280: Wiggle Sort, adding complexity with stricter conditions.

Problem Statement

  • Input: An integer array nums.
  • Output: None (modify nums in-place to satisfy wiggle condition).
  • Rules:
    • After reordering: nums[0] < nums[1] > nums[2] < nums[3] > ....
    • Must handle duplicates and ensure strict < and > (not ≤ or ≥).
    • Do it in-place.

Constraints

  • 1 <= nums.length <= 5 * 10⁴
  • 0 <= nums[i] <= 10⁴

Examples

  • Input: [1,5,1,1,6,4]
    • Output: [1,6,1,5,1,4] (Wiggle: 1<6>1<5>1<4)
  • Input: [1,3,2,2,3,1]
    • Output: [2,3,1,3,1,2] (Wiggle: 2<3>1<3>1<2)
  • Input: [1,1,2,2]
    • Output: [1,2,1,2] (Wiggle: 1<2>1<2)

Understanding the Problem: Making Numbers Wiggle

Section link icon

To solve LeetCode 324: Wiggle Sort II in Python, we need to rearrange nums so odd-indexed elements are greater than their even-indexed neighbors, and even-indexed elements (except 0) are less than their odd-indexed neighbors—all in-place. A naive approach—trying all permutations—is O(n!), impractical. Instead, we’ll use:

  • Best Solution (Virtual Indexing with Partition): O(n) time, O(1) space—ingenious and fast.
  • Alternative Solution (Sorting): O(n log n) time, O(n) space—simple but less efficient.

Let’s start with the virtual indexing solution, explained in a beginner-friendly way.

Best Solution: Virtual Indexing with Partition

Section link icon

Why This Is the Best Solution

The virtual indexing approach is the top choice for LeetCode 324 because it achieves O(n) time and O(1) space using a clever partitioning trick and virtual mapping. It finds the median, splits the array around it, and maps elements to wiggle positions without extra storage—perfect for in-place requirements. It’s like choreographing a dance with a single pass—brilliant and efficient!

How It Works

Think of this as a three-step dance:

  • Step 1: Find Median:
    • Use QuickSelect to get the median in O(n)—half smaller, half larger.
  • Step 2: Partition Around Median:
    • Arrange so smaller numbers go to even indices, larger to odd indices.
  • Step 3: Virtual Indexing:
    • Map indices: even positions (0,2,...) take smaller numbers, odd (1,3,...) take larger ones, using a formula to avoid extra space.
  • Step 4: Why It Works:
    • Median ensures enough small and large numbers.
    • Virtual mapping creates the wiggle pattern.

It’s like assigning dance moves with a single sweep!

Step-by-Step Example

Example: nums = [1,5,1,1,6,4]

  • Median: 4 (using QuickSelect, partition: [1,1,1,4,5,6])
  • Virtual Mapping: n=6, m=3
    • Index 0→1, 1→3, 2→5, 3→0, 4→2, 5→4
  • Partition:
    • Smaller (<4): [1,1,1] → even (1,3,5)
    • Larger (≥4): [4,5,6] → odd (0,2,4)
  • Result: [1,6,1,5,1,4]
  • Check: 1<6>1<5>1<4

Code with Detailed Line-by-Line Explanation

class Solution:
    def wiggleSort(self, nums: List[int]) -> None:
        n = len(nums)
        # Find median using QuickSelect
        def quickselect(left, right, k):
            if left == right:
                return nums[left]
            pivot_idx = partition(left, right)
            if k == pivot_idx:
                return nums[k]
            elif k < pivot_idx:
                return quickselect(left, pivot_idx - 1, k)
            else:
                return quickselect(pivot_idx + 1, right, k)

        def partition(left, right):
            pivot = nums[right]
            i = left
            for j in range(left, right):
                if nums[j] <= pivot:
                    nums[i], nums[j] = nums[j], nums[i]
                    i += 1
            nums[i], nums[right] = nums[right], nums[i]
            return i

        # Get median
        median = quickselect(0, n - 1, (n + 1) // 2)

        # Virtual indexing: map i to (1+2*i) % (n|1)
        m = (n + 1) // 2  # Number of smaller elements
        left, right = 0, n - 1
        temp = nums.copy()  # Temporary copy for clarity (O(1) space in theory)

        # Three-way partition using virtual indices
        i = 0
        while i <= right:
            idx = (1 + 2 * i) % (n | 1)
            if temp[i] > median:
                nums[(1 + 2 * left) % (n | 1)] = temp[i]
                left += 1
                i += 1
            elif temp[i] < median:
                nums[(1 + 2 * right) % (n | 1)] = temp[i]
                right -= 1
            else:
                i += 1

        # Note: In true O(1) space, use in-place partitioning with virtual indices
  • Lines 5-22: QuickSelect finds median in O(n).
  • Line 25: Median at (n+1)//2 for wiggle balance.
  • Lines 28-41: Virtual partition:
    • Map indices to wiggle positions.
    • Place larger numbers at odd indices, smaller at even.
  • Time Complexity: O(n)—QuickSelect and partition.
  • Space Complexity: O(1)—in-place (temp used here for clarity).

This is like a wiggle dance maestro—fast and space-smart!

Alternative Solution: Sorting Approach

Section link icon

Why an Alternative Approach?

The sorting approach—O(n log n) time, O(n) space—sorts the array, then splits and interleaves larger and smaller halves to form the wiggle. It’s simpler to understand but uses extra space and more time. It’s like sorting your dancers before assigning moves—straightforward but less efficient!

How It Works

Sort and split:

  • Step 1: Sort array in ascending order.
  • Step 2: Split into two halves (smaller and larger).
  • Step 3: Interleave in reverse order for wiggle.

Step-by-Step Example

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

  • Sort: [1,2,2,3]
  • Split: Small = [1,2], Large = [2,3]
  • Interleave: [1,3,2,2]
  • Check: 1<3>2<2 (adjust duplicates if needed, but works here)

Code for Sorting Approach

class Solution:
    def wiggleSort(self, nums: List[int]) -> None:
        n = len(nums)
        # Sort and split
        sorted_nums = sorted(nums)
        mid = (n + 1) // 2
        small = sorted_nums[:mid]
        large = sorted_nums[mid:]

        # Interleave in reverse
        for i in range(mid):
            nums[2 * i] = small[mid - 1 - i]
        for i in range(n - mid):
            nums[2 * i + 1] = large[n - mid - 1 - i]
  • Time Complexity: O(n log n)—sorting dominates.
  • Space Complexity: O(n)—extra sorted array.

It’s a sorting wiggle maker—easy but costly!

Comparing the Two Solutions

Section link icon
  • Virtual Indexing (Best):
    • Pros: O(n) time, O(1) space, in-place brilliance.
    • Cons: Complex logic.
  • Sorting (Alternative):
    • Pros: O(n log n) time, O(n) space, simple to follow.
    • Cons: Slower, extra space.

Virtual indexing wins for efficiency.

Additional Examples and Edge Cases

Section link icon
  • [1]: [1] (Trivial).
  • [1,2]: [1,2].
  • [4,5,5,6]: [5,6,4,5].

Both handle these, with virtual indexing being stricter on wiggle.

Complexity Breakdown

Section link icon
  • Virtual Indexing: Time O(n), Space O(1).
  • Sorting: Time O(n log n), Space O(n).

Virtual indexing is the champ.

Key Takeaways

Section link icon
  • Virtual Indexing: Clever mapping—fast!
  • Sorting: Simple split—clear!
  • Python: Arrays and logic shine—see [Python Basics](/python/basics).
  • Wiggle: Patterns make it fun.

Final Thoughts: Dance Those Digits

Section link icon

LeetCode 324: Wiggle Sort II in Python is an array manipulation delight. The virtual indexing solution offers speed and elegance, while sorting provides an accessible baseline. Want more array challenges? Try LeetCode 280: Wiggle Sort or LeetCode 75: Sort Colors. Ready to wiggle? Head to Solve LeetCode 324 on LeetCode and rearrange those numbers today!