LeetCode 60: Permutation Sequence Solution in Python Explained

Problem Statement

Section link icon

LeetCode 60, Permutation Sequence, is a medium-level problem where you’re given two integers n and k. Your task is to return the (k)-th permutation of the sequence [1, 2, ..., n] in lexicographical order. The permutations are numbered from 1 to (n!), and you need to find the specific (k)-th permutation as a string.

Constraints

  • 1 <= n <= 9: Length of sequence is between 1 and 9.
  • 1 <= k <= n!: \(k\) is a valid permutation index.

Example

Input: n = 3, k = 3
Output: "213"
Explanation: Permutations in order:
1. "123"
2. "132"
3. "213" <- 3rd permutation
4. "231"
5. "312"
6. "321"

Input: n = 4, k = 9
Output: "2314"
Explanation: 9th permutation of [1,2,3,4].

Input: n = 1, k = 1
Output: "1"
Explanation: Only permutation is "1".

Understanding the Problem

Section link icon

How do you solve LeetCode 60: Permutation Sequence in Python? For n = 3 and k = 3, find the 3rd permutation of [1,2,3], which is "213". Generating all (n!) permutations and picking the (k)-th is inefficient (O(n!)), so we need a smarter approach. We’ll explore two solutions: a Mathematical Factorial Approach (optimal and primary) and an Alternative with Iterative Permutation (more intuitive but slower). The factorial method uses math to compute the permutation in O(n^2) time.

Solution 1: Mathematical Factorial Approach (Primary)

Section link icon

Explanation

Use factorials to determine each digit of the permutation. For (n) numbers, there are ((n-1)!) permutations starting with each number. Divide (k-1) by ((n-1)!) to find the index of the first digit, remove it, and repeat with the remaining numbers and updated (k).

Here’s a flow for (n = 3, k = 3):

Numbers: [1,2,3], k = 2 (k-1)
(2)! = 2, k = 2, index = 1 -> "2", remaining = [1,3], k = 0
(1)! = 1, k = 0, index = 0 -> "1", remaining = [3], k = 0
(0)! = 1, k = 0, index = 0 -> "3"
Result: "213"
  1. Initialize Numbers and Factorials.
  • List of numbers [1,2,...,n], precompute factorials.
  1. Compute Digits.
  • Use \(k-1\) to find each digit’s index, adjust \(k\).
  1. Build String.
  • Append digits, remove used numbers.
  1. Return Result.

Step-by-Step Example

Example 1: n = 3, k = 3

We need "213".

  • Goal: Find 3rd permutation of [1,2,3].
  • Result for Reference: "213".
  • Start: n = 3, k = 3, numbers = [1,2,3], result = "", k = 2 (k-1).
  • Step 1: n = 3, factorial = (2)! = 2.
    • Index = k // 2 = 2 // 2 = 1, digit = numbers[1] = 2.
    • result = "2", remove 2, numbers = [1,3], k = k % 2 = 0.
  • Step 2: n = 2, factorial = (1)! = 1.
    • Index = 0 // 1 = 0, digit = numbers[0] = 1.
    • result = "21", remove 1, numbers = [3], k = 0 % 1 = 0.
  • Step 3: n = 1, factorial = (0)! = 1.
    • Index = 0 // 1 = 0, digit = numbers[0] = 3.
    • result = "213", numbers = [].
  • Finish: Return "213".

Example 2: n = 4, k = 9

We need "2314".

  • Start: numbers = [1,2,3,4], k = 8 (k-1).
  • Step 1: (3)! = 6, index = 8 // 6 = 1, digit = 2, k = 8 % 6 = 2.
    • result = "2", numbers = [1,3,4].
  • Step 2: (2)! = 2, index = 2 // 2 = 1, digit = 3, k = 2 % 2 = 0.
    • result = "23", numbers = [1,4].
  • Step 3: (1)! = 1, index = 0 // 1 = 0, digit = 1, k = 0.
    • result = "231", numbers = [4].
  • Step 4: (0)! = 1, index = 0, digit = 4.
    • result = "2314", numbers = [].
  • Finish: Return "2314".

How the Code Works (Mathematical Factorial)

Here’s the Python code with line-by-line explanation:

def getPermutation(n: int, k: int) -> str:
    # Line 1: Initialize numbers and factorials
    numbers = list(range(1, n + 1))
    factorials = [1] * (n + 1)
    for i in range(1, n):
        factorials[i] = factorials[i-1] * i

    # Line 2: Adjust k to 0-based index
    k -= 1

    # Line 3: Build permutation
    result = []
    for i in range(n-1, -1, -1):
        if i == 0:
            result.append(str(numbers[0]))
            break
        # Line 4: Calculate index and digit
        index = k // factorials[i]
        k = k % factorials[i]
        digit = numbers[index]
        # Line 5: Add digit and remove from numbers
        result.append(str(digit))
        numbers.pop(index)

    # Line 6: Return permutation as string
    return ''.join(result)

Alternative: Iterative Permutation Approach

Section link icon

Explanation

Generate permutations iteratively using the "next permutation" algorithm (like LeetCode 31), starting from [1,2,...,n] and iterating until the (k)-th permutation is reached. This is less efficient but mimics the lexicographical order directly.

  1. Initialize Sequence.
  2. Generate Next Permutation.
  • Swap and reverse to get next in order.

3. Iterate (k-1) Times. 4. Return Result.

Step-by-Step Example (Alternative)

For (n = 3, k = 3):

  • Start: nums = [1,2,3], k = 3.
  • Step 1: 1st: [1,2,3].
  • Step 2: 2nd: [1,3,2] (swap 2,3, reverse nothing).
  • Step 3: 3rd: [2,1,3] (swap 1,3, reverse [3,2]).
  • Finish: Return "213".

How the Code Works (Iterative Permutation)

def getPermutationIter(n: int, k: int) -> str:
    # Line 1: Initialize sequence
    nums = list(range(1, n + 1))

    # Line 2: Generate k-1 next permutations
    for _ in range(k - 1):
        # Line 3: Find first decreasing element from right
        i = n - 2
        while i >= 0 and nums[i] >= nums[i + 1]:
            i -= 1
        # Line 4: Find number to swap
        j = n - 1
        while j >= 0 and nums[j] <= nums[i]:
            j -= 1
        # Line 5: Swap
        nums[i], nums[j] = nums[j], nums[i]
        # Line 6: Reverse after i
        left, right = i + 1, n - 1
        while left < right:
            nums[left], nums[right] = nums[right], nums[left]
            left += 1
            right -= 1

    # Line 7: Return as string
    return ''.join(map(str, nums))

Complexity

  • Mathematical Factorial:
    • Time: O(n^2) – n iterations, O(n) for list removal.
    • Space: O(n) – Numbers list and factorials.
  • Iterative Permutation:
    • Time: O(k * n) – k iterations, O(n) per next permutation, up to O(n * n!).
    • Space: O(n) – Store sequence.

Efficiency Notes

Mathematical Factorial is O(n^2) time and O(n) space, optimal for LeetCode 60 as it avoids generating all permutations. Iterative Permutation is O(k * n) time, potentially O(n * n!), making it impractical for large (k), but it’s a clear alternative.

Key Insights

Tips to master LeetCode 60:

  • Factorial Math: Use \((n-1)!\) to find each digit.
  • Index Adjustment: \(k-1\) for 0-based indexing.
  • Efficient Build: Avoid full permutation generation.

Additional Example

Section link icon

For (n = 2, k = 2):

  • Goal: "21".
  • Factorial: [1,2], k-1=1, (1)! = 1, index = 1, "2", "1" -> "21".
  • Result: "21".

Edge Cases

Section link icon
  • \(n = 1, k = 1\): "1".
  • \(n = 9, k = 362880\): Last permutation of [1,2,3,4,5,6,7,8,9].
  • \(n = 2, k = 1\): "12".

Final Thoughts

Section link icon

The Mathematical Factorial solution is a brilliant pick for LeetCode 60 in Python—fast, clever, and math-driven. For a related challenge, try LeetCode 59: Spiral Matrix II to switch to matrix generation! Ready to permute? Solve LeetCode 60 on LeetCode now and test it with (n=3, k=3) or (n=4, k=9) to master permutation sequences. Dive in and sequence it up!