LeetCode 390: Elimination Game Solution in Python – A Step-by-Step Guide

Imagine you’re playing a game with a circle of numbers from 1 to n—like 1 to 5—and you eliminate every other number, alternating directions (left-to-right, then right-to-left), until only one remains. That’s the challenge of LeetCode 390: Elimination Game, a medium-level problem that’s all about finding the last number standing through a pattern. Using Python, we’ll explore two solutions: the Best Solution—mathematical pattern recognition for O(log n) efficiency—and an Alternative Solution—simulation with a list at O(n²). With examples, clear code breakdowns, and a friendly vibe, this guide will help you find that survivor. Let’s start eliminating!

What Is LeetCode 390: Elimination Game?

Section link icon

LeetCode 390: Elimination Game gives you an integer n, representing numbers from 1 to n arranged in a list. You eliminate every other number, starting left-to-right, then right-to-left, repeating until one number remains. Your task is to return that last number. It’s a problem that tests your ability to identify patterns or simulate eliminations efficiently!

Problem Statement

  • Input:
    • n: Integer representing the number of elements (1 to n).
  • Output: Integer - The last remaining number.
  • Rules:
    • Start with list [1, 2, ..., n].
    • Eliminate every other number (e.g., 1st pass: remove even indices).
    • Alternate direction: left-to-right, then right-to-left, repeat.
    • Continue until one number remains.

Constraints

  • 1 <= n <= 10⁹

Examples

  • Input: n = 9
    • Output: 6
      • [1,2,3,4,5,6,7,8,9] → [2,4,6,8] (left-to-right).
      • [2,4,6,8] → [2,6] (right-to-left).
      • [2,6] → [6] (left-to-right).
  • Input: n = 6
    • Output: 4
      • [1,2,3,4,5,6] → [2,4,6] → [2,4] → [4].
  • Input: n = 1
    • Output: 1
      • [1] → [1].

These show the elimination dance—let’s solve it!

Understanding the Problem: Finding the Last Number

Section link icon

To tackle LeetCode 390 in Python, we need to: 1. Simulate or deduce the elimination process from 1 to n. 2. Alternate directions (left-to-right, right-to-left). 3. Find the last remaining number efficiently.

A naive approach might simulate with a list, but that’s O(n²) and slow for n up to 10⁹. We’ll use:

  • Best Solution: Mathematical pattern (Josephus variant)—O(log n) time, O(1) space—fast and optimal.
  • Alternative Solution: Simulation with list—O(n²) time, O(n) space—intuitive but impractical.

Let’s deduce with the best solution!

Best Solution: Mathematical Pattern (Josephus Variant)

Section link icon

Why This Is the Best Solution

The mathematical pattern approach runs in O(log n) time with O(1) space by recognizing a recursive pattern in the elimination process, akin to the Josephus problem but with alternating directions. It’s the most efficient—avoiding simulation entirely and computing the result directly, making it ideal for large n!

How It Works

Think of it like a shrinking circle:

  • Pattern:
    • Left-to-right: Eliminate every 2nd number, survivor shifts based on remaining count.
    • Right-to-left: Adjust based on parity and remaining numbers.
  • Key Insight:
    • For each step, halve the count (n //= 2).
    • Track first number (head), update with step size (2^k).
    • Direction alternates, affecting head adjustment.
  • Formula:
    • If n is odd and left-to-right, head stays or shifts.
    • Recursively compute until n = 1.
  • Result: Final head is the answer.

It’s like decoding the survivor’s position step-by-step!

Step-by-Step Example

For n = 9:

  • Init: head = 1, n = 9, step = 1, left = True.
  • Pass 1 (left-to-right):
    • n = 9, odd, eliminate every 2nd: [2,4,6,8].
    • head = 1 + step = 2, n = 9 // 2 = 4, step *= 2 = 2, left = False.
  • Pass 2 (right-to-left):
    • n = 4, even, [2,4,6,8] → [2,6].
    • head = 2 (no shift for even right-to-left), n = 4 // 2 = 2, step *= 2 = 4, left = True.
  • Pass 3 (left-to-right):
    • n = 2, even, [2,6] → [2].
    • head = 2 + step = 6, n = 2 // 2 = 1, step *= 2 = 8, left = False.
  • n = 1: Return head = 6.
  • Result: 6.

For n = 6:

  • Init: head = 1, n = 6, step = 1.
  • Pass 1: head = 2, n = 3, step = 2, left = False.
  • Pass 2: head = 2 + 2 = 4, n = 1, step = 4, left = True.
  • Result: 4.

Code with Explanation

Here’s the Python code using the mathematical pattern:

class Solution:
    def lastRemaining(self, n: int) -> int:
        # Initialize head, step, and direction
        head = 1
        step = 1
        left = True

        # Eliminate until one remains
        while n > 1:
            # Left-to-right or odd count from right
            if left or n % 2 == 1:
                head += step
            # Halve the count, double the step
            n //= 2
            step *= 2
            left = not left

        return head

Explanation

  • Lines 4-6: head: First number, step: Elimination step, left: Direction.
  • Lines 8-15: Loop until n = 1:
    • If left-to-right or n odd (right-to-left), increment head by step—O(1).
    • Halve n, double step, flip direction—O(1).
  • Line 17: Return final head—O(1).
  • Time: O(log n)—halves n each iteration.
  • Space: O(1)—few variables.

This is like a pattern decoder—swift and sleek!

Alternative Solution: Simulation with List

Section link icon

Why an Alternative Approach?

The simulation solution runs in O(n²) time with O(n) space by explicitly eliminating numbers in a list, alternating directions. It’s intuitive—great for small n or understanding the process, but impractical for large n due to time complexity!

How It Works

  • List: Initialize with 1 to n.
  • Simulate:
    • Left-to-right: Remove every 2nd element.
    • Right-to-left: Reverse and repeat.
  • Repeat: Until one remains.
  • Result: Last number.

Step-by-Step Example

For n = 6:

  • Init: List = [1, 2, 3, 4, 5, 6].
  • Left-to-right: [2, 4, 6].
  • Right-to-left: [2, 4] (remove 6).
  • Left-to-right: [4].
  • Result: 4.

Code with Explanation

class Solution:
    def lastRemaining(self, n: int) -> int:
        # Initialize list
        arr = list(range(1, n + 1))
        left = True

        # Eliminate until one remains
        while len(arr) > 1:
            if left:
                arr = arr[1::2]  # Every other from left
            else:
                arr = arr[-2::-2] if len(arr) % 2 == 0 else arr[-1::-2]  # From right
            left = not left

        return arr[0]

Explanation

  • Line 4: arr: List of 1 to n—O(n).
  • Line 5: left: Direction flag.
  • Lines 7-12: Eliminate:
    • Left: Take every 2nd element—O(n).
    • Right: Reverse slice based on parity—O(n).
  • Line 14: Return last element—O(1).
  • Time: O(n²)—n eliminations, O(n) per step.
  • Space: O(n)—list.

It’s a brute-force eliminator—clear but slow!

Comparing the Two Solutions

Section link icon
  • Mathematical Pattern (Best):
    • Time: O(log n)—logarithmic.
    • Space: O(1)—minimal.
    • Pros: Fast, space-efficient, scalable.
    • Cons: Pattern-based, less intuitive.
  • Simulation (Alternative):
    • Time: O(n²)—quadratic.
    • Space: O(n)—list.
    • Pros: Easy to follow.
    • Cons: Slow, memory-heavy.

Math pattern wins for efficiency!

Additional Examples and Edge Cases

Section link icon
  • n=1: Both return 1.
  • n=2: Both return 2.
  • Large n: Math scales, simulation lags.

Math is optimal, simulation works for small n.

Complexity Breakdown

Section link icon
  • Math Pattern:
    • Time: O(log n).
    • Space: O(1).
  • Simulation:
    • Time: O(n²).
    • Space: O(n).

Math excels for performance.

Key Takeaways

Section link icon
  • Math Pattern: Decode the survivor—fast and lean!
  • Simulation: Play it out—clear but slow!
  • Elimination: Patterns beat brute force.
  • Python Tip: Logic rocks—see [Python Basics](/python/basics).

Final Thoughts: Find That Survivor

Section link icon

LeetCode 390: Elimination Game in Python is a pattern challenge. The mathematical approach is the efficiency champ, while simulation offers a tangible alternative for small cases. Want more? Try LeetCode 62: Unique Paths or LeetCode 384: Shuffle an Array. Ready to eliminate? Visit Solve LeetCode 390 on LeetCode and find that last number today!