LeetCode 114: Flatten Binary Tree to Linked List Solution in Python – A Step-by-Step Guide

Picture a binary tree—like a family tree with parents and kids—sprawling out with nodes like [1, 2, 5, 3, 4, None, 6]. Now imagine squashing it into a single, straight line, like a chain, where each node points only to the right: [1, None, 2, None, 3, None, 4, None, 5, None, 6]. That’s the challenge of LeetCode 114: Flatten Binary Tree to Linked List, a medium-level problem that’s all about reshaping a tree in-place. Using Python, we’ll tackle it with two clever solutions: the Best Solution, a Morris-like preorder traversal that uses O(1) space, and an Alternative Solution, a recursive preorder approach with O(n) space. With detailed examples, code breakdowns, and a friendly tone, this guide will help you flatten that tree—whether you’re gearing up for an interview or just love tree puzzles. Let’s get to work and straighten things out!

What Is LeetCode 114: Flatten Binary Tree to Linked List?

Section link icon

In LeetCode 114: Flatten Binary Tree to Linked List, you’re given the root of a binary tree, and your job is to transform it into a “linked list” in-place. The catch? The list must follow the preorder traversal order (root, left, right), and each node’s left child should be None, with its right child pointing to the next node in the sequence. For example, a tree like [1, 2, 5, 3, 4, None, 6] becomes a right-skewed list where 1 points to 2, 2 to 3, 3 to 4, 4 to 5, and 5 to 6—all with no left children.

Problem Statement

  • Input: Root node of a binary tree (TreeNode).
  • Output: None (modify the tree in-place).
  • Rules:
    • Flatten to a linked list using preorder traversal.
    • Set each node’s left to None, right to the next node.
    • Do it in-place (minimize extra space).

Constraints

  • Number of nodes: 0 <= n <= 2000
  • -100 <= Node.val <= 100

Examples

  • Input: root = [1, 2, 5, 3, 4, None, 6]
    • Output: [1, None, 2, None, 3, None, 4, None, 5, None, 6]
    • Why: Preorder: 1, 2, 3, 4, 5, 6 → linked list.
  • Input: root = []
    • Output: []
    • Why: Empty tree, nothing to flatten.
  • Input: root = [0]
    • Output: [0]
    • Why: Single node is already a list.

Understanding the Problem: From Tree to Chain

Section link icon

To solve LeetCode 114: Flatten Binary Tree to Linked List in Python, we need to rearrange a binary tree into a right-skewed linked list following preorder traversal, all while modifying the tree in-place. A naive approach might be to collect all nodes in a list (O(n) space) and relink them, but the “in-place” hint pushes us toward minimal extra space. We’ll explore:

  • Best Solution (Morris-like Preorder): O(n) time, O(1) space—rewires the tree on the fly.
  • Alternative Solution (Recursive Preorder): O(n) time, O(n) space—uses the call stack.

Let’s dive into the Best Solution—a Morris-like approach—and break it down step-by-step.

Best Solution: Morris-like Preorder Traversal

Section link icon

Why This Is the Best Solution

The Morris-like preorder traversal is the champ here because it flattens the tree in O(n) time with O(1) space—no recursion, no extra arrays, just clever pointer juggling. It reuses the tree’s own structure to build the list, making it memory-efficient and elegant. It’s like a magician reweaving the tree into a chain without needing extra props!

How It Works

Here’s the strategy:

  • Step 1: Process Node-by-Node:
    • Start at the root, work iteratively.
  • Step 2: Handle Left Subtree:
    • If a node has a left child:
      • Find the rightmost node in the left subtree (the “predecessor”).
      • Move the left subtree to the right of the current node.
      • Set the current node’s left to None.
      • Link the predecessor’s right to the original right child.
  • Step 3: Move Forward:
    • Advance to the new right child (former left subtree).
  • Step 4: Repeat:
    • Continue until all nodes are processed.

Step-by-Step Example

Example: root = [1, 2, 5, 3, 4, None, 6]

  • Initial Tree:

1 / \ 2 5 / \ \ 3 4 6

  • Step 1: root = 1:
    • Left = 2, Right = 5.
    • Find rightmost in left (2’s subtree) = 4.
    • 1.right = 2, 4.right = 5, 1.left = None.
    • Tree:

1 \ 2 / \ 3 4 \ 5 \ 6

  • Step 2: root = 2:
    • Left = 3, Right = 4.
    • Rightmost in left = 3 (no right child).
    • 2.right = 3, 3.right = 4, 2.left = None.
    • Tree:

1 \ 2 \ 3 \ 4 \ 5 \ 6

  • Step 3: root = 3:
    • No left, move to 4.
  • Step 4: root = 4:
    • No left, move to 5.
  • Step 5: root = 5:
    • No left, move to 6.
  • Step 6: root = 6:
    • No left, done.
  • Result: [1, None, 2, None, 3, None, 4, None, 5, None, 6].

Code with Detailed Line-by-Line Explanation

Here’s the Python code, explained clearly:

class Solution:
    def flatten(self, root: Optional[TreeNode]) -> None:
        # Start at root
        curr = root

        # Iterate until no nodes left
        while curr:
            # If no left child, move to right
            if not curr.left:
                curr = curr.right
            else:
                # Find rightmost node in left subtree
                pred = curr.left
                while pred.right:
                    pred = pred.right

                # Rewire: left subtree to right, original right to end
                pred.right = curr.right
                curr.right = curr.left
                curr.left = None
                # Move to next node (former left)
                curr = curr.right
  • Line 4: Start with curr at root.
  • Line 7-19: Loop over nodes:
    • Line 9-10: No left child, skip to right.
    • Line 12-14: Find predecessor (rightmost in left subtree).
    • Line 17-18: Link predecessor’s right to current right, move left subtree to right, clear left.
    • Line 20: Advance to new right.
  • Time Complexity: O(n)—each node visited once.
  • Space Complexity: O(1)—no extra space.

This rewires the tree like a pro!

Alternative Solution: Recursive Preorder Traversal

Section link icon

Why an Alternative Approach?

The recursive preorder approach is simpler to grasp—O(n) time, O(n) space via the call stack. It collects the preorder sequence implicitly and flattens as it unwinds. It’s like writing down the order first, then relinking—intuitive but less space-efficient.

How It Works

Here’s the plan:

  • Step 1: Use recursion to process preorder (root, left, right).
  • Step 2: Track the previous node globally.
  • Step 3: Link each node’s right to the next, set left to None.

Step-by-Step Example

Example: root = [1, 2, 5, 3, 4, None, 6]

  • Preorder: 1, 2, 3, 4, 5, 6.
  • Recursion:
    • 1: prev = None, link to 2, prev = 1.
    • 2: Link to 3, prev = 2.
    • 3: Link to 4, prev = 3.
    • 4: Link to 5, prev = 4.
    • 5: Link to 6, prev = 5.
    • 6: Done.
  • Result: [1, None, 2, None, 3, None, 4, None, 5, None, 6].

Code for Recursive Approach

class Solution:
    def __init__(self):
        self.prev = None

    def flatten(self, root: Optional[TreeNode]) -> None:
        if not root:
            return

        # Store right child (will process later)
        right = root.right

        # Link previous to current, clear left
        if self.prev:
            self.prev.right = root
            self.prev.left = None
        self.prev = root

        # Recurse on left, then right
        self.flatten(root.left)
        self.flatten(right)
  • Line 3-4: Global prev to track last node.
  • Line 8-18: Recursive function:
    • Line 11: Save right child.
    • Line 14-15: Link prev to current, update prev.
    • Line 18-19: Process left, then right.
  • Time Complexity: O(n)—each node visited once.
  • Space Complexity: O(n)—recursion stack.

It’s a recursive chain builder!

Comparing the Two Solutions

Section link icon
  • Morris-like (Best):
    • Pros: O(n) time, O(1) space, in-place magic.
    • Cons: Trickier to follow.
  • Recursive (Alternative):
    • Pros: O(n) time, O(n) space, simpler logic.
    • Cons: Uses stack space.

Morris-like wins for efficiency.

Additional Examples and Edge Cases

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

Both handle these seamlessly.

Complexity Breakdown

Section link icon
  • Morris-like: Time O(n), Space O(1).
  • Recursive: Time O(n), Space O(n).

Morris-like is leaner.

Key Takeaways

Section link icon
  • Morris-like: Rewire with no space!
  • Recursive: Preorder with ease!
  • In-Place: Pointer play is key.
  • Python Tip: Classes help globals—see [Python Basics](/python/basics).

Final Thoughts: Straighten That Tree

Section link icon

LeetCode 114: Flatten Binary Tree to Linked List in Python is a tree-twisting treat. The Morris-like approach dazzles with O(1) space, while recursive preorder offers clarity. Want more tree fun? Try LeetCode 94: Binary Tree Inorder Traversal or LeetCode 102: Binary Tree Level Order Traversal. Ready to flatten? Head to Solve LeetCode 114 on LeetCode and reshape that tree today!