LeetCode 354: Russian Doll Envelopes Solution in Python – A Step-by-Step Guide

Imagine you’re handed a stack of envelopes—like [[5,4], [6,4], [6,7], [2,3]]—each with a width and height, and your challenge is to figure out how many you can nest inside one another, like Russian dolls, where each envelope must be strictly smaller in both dimensions than the one containing it. This is LeetCode 354: Russian Doll Envelopes, a hard-level problem that blends sorting, dynamic programming, and optimization. Using Python, we’ll explore two powerful solutions: the Best Solution, a dynamic programming approach with binary search that’s efficient at O(n log n) time, and an Alternative Solution, a standard DP approach running in O(n²) time. With detailed examples, code breakdowns, and a friendly tone, this guide will help you stack those envelopes—whether you’re new to coding or prepping for a tough interview. Let’s dive in and start nesting!

What Is LeetCode 354: Russian Doll Envelopes?

Section link icon

In LeetCode 354: Russian Doll Envelopes, you’re given an array envelopes, where envelopes[i] = [w_i, h_i] represents the width and height of the i-th envelope. Your task is to find the maximum number of envelopes that can be nested, where an envelope can fit inside another only if both its width and height are strictly less than the outer envelope’s dimensions (w_i < w_j and h_i < h_j). For example, with envelopes = [[5,4], [6,4], [6,7], [2,3]], the maximum nesting is 3 (e.g., [2,3] → [5,4] → [6,7]).

Problem Statement

  • Input: An array envelopes of integer pairs [w_i, h_i].
  • Output: An integer—the maximum number of envelopes that can be nested.
  • Rules:
    • Envelope i fits in j if w_i < w_j and h_i < h_j.
    • Return the length of the longest nesting sequence.

Constraints

  • 1 <= envelopes.length <= 10⁵
  • envelopes[i].length == 2
  • 1 <= w_i, h_i <= 10⁵

Examples

  • Input: envelopes = [[5,4], [6,4], [6,7], [2,3]]
    • Output: 3
    • Why: [2,3] → [5,4] → [6,7] (or [2,3] → [6,4] → [6,7]).
  • Input: envelopes = [[1,1], [1,1], [1,1]]
    • Output: 1
    • Why: All identical, only one can be used.
  • Input: envelopes = [[4,5], [4,6], [6,7], [2,3], [1,1]]
    • Output: 4
    • Why: [1,1] → [2,3] → [4,5] → [6,7].

Understanding the Problem: Nesting the Envelopes

Section link icon

To solve LeetCode 354: Russian Doll Envelopes in Python, we need to find the longest sequence of envelopes where each fits inside the next, based on strict width and height comparisons. This is a 2D version of the Longest Increasing Subsequence (LIS) problem. A naive approach—trying all subsequences—would take O(2^n) time, which is impractical for 10⁵ envelopes. Instead, we’ll optimize with:

  • Best Solution (DP with Binary Search): O(n log n) time, O(n) space—fast and efficient.
  • Alternative Solution (Standard DP): O(n²) time, O(n) space—simpler but slower.

Let’s dive into the Best Solution—DP with binary search—and break it down step-by-step.

Alternative Solution: Standard Dynamic Programming

Section link icon

Why an Alternative Approach?

The standard DP approach offers a simpler, more intuitive take—O(n²) time, O(n) space—using a classic LIS method with a DP array to track the longest sequence ending at each envelope. It’s easier to understand but less efficient for large inputs. It’s like stacking envelopes one by one, checking all previous ones to see what fits—straightforward but slower!

How It Works

  • Step 1: Sort by width ascending, height ascending.
  • Step 2: DP array dp[i]: max nesting ending at envelope i.
  • Step 3: Fill DP by comparing with previous envelopes.
  • Step 4: Return max DP value.

Step-by-Step Example

Example: envelopes = [[5,4], [6,4], [6,7], [2,3]]

  • Step 1: Sort:
    • [[2,3], [5,4], [6,4], [6,7]].
  • Step 2: Initialize DP:
    • dp = [1, 1, 1, 1] (each starts with 1).
  • Step 3: Fill DP:
    • i=0: [2,3], dp[0] = 1.
    • i=1: [5,4], w>h of [2,3], dp[1] = 1 + 1 = 2.
    • i=2: [6,4], w>h of [2,3], dp[2] = 1 + 1 = 2.
    • i=3: [6,7], w>h of [2,3], [5,4], dp[3] = 1 + 2 = 3.
  • Step 4: Result:
    • max(dp) = 3.
  • Result: 3.

Code for DP Approach

class Solution:
    def maxEnvelopes(self, envelopes: List[List[int]]) -> int:
        # Step 1: Sort by width and height ascending
        envelopes.sort(key=lambda x: (x[0], x[1]))

        # Step 2: Initialize DP array
        n = len(envelopes)
        dp = [1] * n

        # Step 3: Fill DP
        for i in range(1, n):
            for j in range(i):
                if envelopes[i][0] > envelopes[j][0] and envelopes[i][1] > envelopes[j][1]:
                    dp[i] = max(dp[i], dp[j] + 1)

        # Step 4: Return max
        return max(dp)
  • Line 4: Sort envelopes.
  • Line 7-8: Each envelope starts with length 1.
  • Line 11-14: Compare with previous envelopes.
  • Line 17: Return longest sequence.
  • Time Complexity: O(n²)—n comparisons per n.
  • Space Complexity: O(n)—DP array.

It’s a classic nesting tracker!

Comparing the Two Solutions

Section link icon
  • DP with Binary Search (Best):
    • Pros: O(n log n) time, O(n) space, fast and scalable.
    • Cons: Requires binary search understanding.
  • Standard DP (Alternative):
    • Pros: O(n²) time, O(n) space, simpler logic.
    • Cons: Slower for large n.

Binary search wins for efficiency.

Additional Examples and Edge Cases

Section link icon
  • [[1,1]]: 1.
  • [[1,2], [2,3], [3,1]]: 2.
  • [[2,1], [2,2]]: 1.

Both handle these perfectly.

Complexity Breakdown

Section link icon
  • Binary Search: Time O(n log n), Space O(n).
  • Standard DP: Time O(n²), Space O(n).

Binary search’s speed shines.

Key Takeaways

Section link icon
  • Binary Search: Sort and search smart!
  • Standard DP: Compare all pairs!
  • Nesting: 2D LIS twist.
  • Python Tip: Sorting simplifies—see [Python Basics](/python/basics).

Final Thoughts: Stack the Dolls

Section link icon

LeetCode 354: Russian Doll Envelopes in Python is a nesting challenge with depth. The binary search DP stacks efficiently, while standard DP offers clarity. Want more sequence fun? Try LeetCode 300: Longest Increasing Subsequence or LeetCode 1143: Longest Common Subsequence. Ready to nest? Head to Solve LeetCode 354 on LeetCode and stack those envelopes today!