LeetCode 26: Remove Duplicates from Sorted Array Solution in Python Explained

Problem Statement

Section link icon

LeetCode 26, Remove Duplicates from Sorted Array, is an easy-level problem where you’re given a sorted array of integers nums. Your task is to remove duplicates in-place and return the length of the new array containing only unique elements. The array is sorted in ascending order, and you must modify it such that all duplicates are excluded from the first k elements, where k is the number of unique elements. The elements beyond the first k positions don’t matter.

Constraints

  • 0 <= nums.length <= 3 * 10^4: Array length ranges from 0 to 30,000.
  • -100 <= nums[i] <= 100: Each element is between -100 and 100.
  • Array is sorted in ascending order.

Example

Input: nums = [1,1,2]
Output: 2, nums = [1,2,_]
Explanation: Unique elements are [1,2], length = 2. "_" can be any value.

Input: nums = [0,0,1,1,1,2,2,3,3,4]
Output: 5, nums = [0,1,2,3,4,_,_,_,_,_]
Explanation: Unique elements are [0,1,2,3,4], length = 5.

Input: nums = []
Output: 0, nums = []
Explanation: Empty array has length 0.

Understanding the Problem

Section link icon

How do you solve LeetCode 26: Remove Duplicates from Sorted Array in Python? Given a sorted array like [1,1,2], you need to modify it in-place to [1,2,_] and return 2, the count of unique elements. Since the array is sorted, duplicates are adjacent, making it easier to identify and skip them. We’ll explore two approaches: a Two-Pointer Iterative Solution (optimal and primary) and an In-Place Modification with Comparison (alternative for understanding). Both operate in-place, leveraging the sorted nature of the array.

Solution 1: Two-Pointer Iterative Approach (Primary)

Section link icon

Explanation

Use two pointers: one to track the position for the next unique element (k) and another to scan the array (i). Since the array is sorted, compare adjacent elements—if they differ, place the new unique element at k and increment k.

Here’s a visual for [0,0,1,1,2]:

Initial: [0,0,1,1,2], k = 1, i = 1
Step 1: [0,0,1,1,2], k = 2, i = 2 (0≠1, place 1)
Step 2: [0,1,1,1,2], k = 3, i = 4 (1≠2, place 2)
Final: [0,1,2,1,2], return k = 3
  1. Handle Edge Case.
  • If array is empty, return 0.
  1. Initialize Pointers.
  • k = 1: Position for next unique element (start after first element).
  • i = 1: Scanner starting from second element.
  1. Scan and Place Unique Elements.
  • Compare nums[i] with nums[i-1]. If different, copy nums[i] to nums[k] and increment k.
  1. Return Length.
  • Return k, the count of unique elements.

Step-by-Step Example

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

We need length 5, nums = [0,1,2,3,4,,,,,_].

  • Goal: Remove duplicates, keep unique elements in order.
  • Result for Reference: Length = 5, nums = [0,1,2,3,4,...].
  • Start: nums = [0,0,1,1,1,2,2,3,3,4], k = 1, i = 1.
  • Step 1: Check edge case.
    • Array not empty, proceed.
  • Step 2: i = 1, compare nums[1] = 0 with nums[0] = 0.
    • Same, skip, i += 1.
  • Step 3: i = 2, compare nums[2] = 1 with nums[1] = 0.
    • Different, nums[k] = nums[2], nums[1] = 1, k = 2, i = 3.
    • nums = [0,1,1,1,1,2,2,3,3,4].
  • Step 4: i = 3, compare nums[3] = 1 with nums[2] = 1.
    • Same, skip, i += 1.
  • Step 5: i = 4, compare nums[4] = 1 with nums[3] = 1.
    • Same, skip, i += 1.
  • Step 6: i = 5, compare nums[5] = 2 with nums[4] = 1.
    • Different, nums[2] = 2, k = 3, i = 6.
    • nums = [0,1,2,1,1,2,2,3,3,4].
  • Step 7: i = 6, compare nums[6] = 2 with nums[5] = 2.
    • Same, skip, i += 1.
  • Step 8: i = 7, compare nums[7] = 3 with nums[6] = 2.
    • Different, nums[3] = 3, k = 4, i = 8.
    • nums = [0,1,2,3,1,2,2,3,3,4].
  • Step 9: i = 8, compare nums[8] = 3 with nums[7] = 3.
    • Same, skip, i += 1.
  • Step 10: i = 9, compare nums[9] = 4 with nums[8] = 3.
    • Different, nums[4] = 4, k = 5, i = 10.
    • nums = [0,1,2,3,4,2,2,3,3,4].
  • Step 11: i = 10, out of bounds, stop.
  • Finish: Return k = 5, nums = [0,1,2,3,4,...].

How the Code Works (Two-Pointer)

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

def removeDuplicates(nums: list[int]) -> int:
    # Line 1: Check if array is empty
    if not nums:
        # Line 2: Return 0 for empty array
        return 0

    # Line 3: Initialize k for unique element position
    k = 1
    # Line 4: Start scanning from second element
    for i in range(1, len(nums)):
        # Line 5: Compare current with previous
        if nums[i] != nums[i-1]:
            # Line 6: Place unique element at k
            nums[k] = nums[i]
            # Line 7: Increment k
            k += 1

    # Line 8: Return count of unique elements
    return k

Alternative: In-Place Modification with Comparison

Section link icon

Explanation

This approach uses a single pointer but explicitly shifts elements leftward to overwrite duplicates, though it’s less efficient due to unnecessary assignments. It’s included to deepen understanding.

  1. Handle Edge Case.
  • Return 0 if empty.
  1. Track Unique Position.
  • Use k to mark where the next unique element goes.
  1. Compare and Shift.
  • Compare with previous unique element; if different, place it at k.
  1. Return Length.

Step-by-Step Example (Alternative)

For nums = [1,1,2]:

  • Goal: Length = 2, nums = [1,2,_].
  • Result for Reference: Length = 2, nums = [1,2,...].
  • Start: nums = [1,1,2], k = 1, i = 1.
  • Step 1: i = 1, nums[1] = 1, nums[0] = 1.
    • Same, skip, i += 1.
  • Step 2: i = 2, nums[2] = 2, nums[0] = 1.
    • Different, nums[1] = 2, k = 2, i = 3.
    • nums = [1,2,2].
  • Step 3: i = 3, out of bounds, stop.
  • Finish: Return k = 2.

How the Code Works (Alternative)

def removeDuplicatesAlternative(nums: list[int]) -> int:
    # Line 1: Check empty array
    if not nums:
        return 0

    # Line 2: Start with first element as unique
    k = 1
    # Line 3: Compare with first unique element initially
    for i in range(1, len(nums)):
        # Line 4: If different from last unique
        if nums[i] != nums[k-1]:
            # Line 5: Place at k
            nums[k] = nums[i]
            # Line 6: Move k forward
            k += 1

    # Line 7: Return unique count
    return k

Complexity

  • Two-Pointer:
    • Time: O(n) – Single pass through array.
    • Space: O(1) – In-place modification.
  • Alternative:
    • Time: O(n) – Single pass.
    • Space: O(1) – In-place.

Efficiency Notes

Both solutions are O(n) time and O(1) space, but the Two-Pointer approach is more intuitive and aligns with LeetCode’s expectations due to its simplicity and minimal writes.

Key Insights

Tips to master LeetCode 26:

  • Sorted Advantage: Use the sorted property—duplicates are adjacent!
  • Two Pointers Shine: k for placement, i for scanning—keep it simple.
  • In-Place Matters: Modify without extra arrays for efficiency.

Additional Example

Section link icon

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

  • Goal: Length = 3, nums = [1,2,3,_,_].
  • Two-Pointer: k = 1, i = 1, skip 1s, place 2 at k = 1, 3 at k = 2, return 3.
  • Result: [1,2,3,2,3], length = 3.

Edge Cases

Section link icon
  • Empty Array: [] – Return 0.
  • No Duplicates: [1,2,3] – Return 3.
  • All Duplicates: [1,1,1] – Return 1, nums = [1,_,_].

Final Thoughts

Section link icon

The Two-Pointer approach is the go-to solution for LeetCode 26 in Python—fast, clean, and beginner-friendly. For a related challenge, explore LeetCode 27: Remove Element next! Try tweaking the code with different inputs to solidify your understanding.