LeetCode 156: Binary Tree Upside Down Solution in Python Explained

Turning a binary tree upside down might feel like flipping a family tree on its head, and LeetCode 156: Binary Tree Upside Down is a medium-level challenge that makes it fascinating! Given the root of a binary tree where all right nodes are either leaf nodes with no children or null, you need to transform it so the leftmost node becomes the new root, with original parents becoming right children and siblings becoming left children, returning the new root. In this blog, we’ll solve it with Python, exploring two solutions—Iterative Left-Rotation (our best solution) and Recursive Left-Rotation (a practical alternative). With step-by-step examples, detailed code breakdowns, and tips, you’ll master this problem. Let’s flip that tree!

Problem Statement

Section link icon

In LeetCode 156, you’re given the root of a binary tree with a specific property: all right nodes are either leaf nodes (no children) or null, and left nodes form a chain leading to the leftmost node. Your task is to transform the tree by making the leftmost node the new root, adjusting pointers so original parents become right children and right siblings become left children, returning the new root. This differs from stack design like LeetCode 155: Min Stack, focusing on tree restructuring rather than data structure operations.

Constraints

  • Number of nodes: 0 <= n <= 10^5
  • Node values: -1000 <= Node.val <= 1000
  • Right nodes are leaves or null

Example

Let’s see some cases:

Input: root = [1,2,3,4,5]
      1
     / \
    2   3
   /
  4
 /
5
Output: [5,4,2,null,null,3,1]
      5
     / \
    4   null
   / \
  2   null
 / \
3   1
Explanation: Leftmost 5 becomes root, 4→right of 5, 2→right of 4, etc.
Input: root = []
Output: null
Explanation: Empty tree, return null.
Input: root = [1,2,null,3]
      1
     /
    2
   /
  3
Output: [3,2,null,1]
      3
     /
    2
   /
  1
Explanation: 3 becomes root, 2→right of 3, 1→left of 2.

These examples show we’re flipping the tree upside down.

Understanding the Problem

Section link icon

How do you solve LeetCode 156: Binary Tree Upside Down in Python? Take root = [1,2,3,4,5]. Originally structured as a left-leaning tree with right leaves, we need 5 as the new root, 4 as its right child, 2 as 4’s right child, and so on, adjusting pointers: 5→4→2→3→1. For [1,2,null,3], 3 becomes the root. This isn’t an array search like LeetCode 154: Find Minimum in Rotated Sorted Array II; it’s about tree transformation. We’ll use: 1. Iterative Left-Rotation: O(n) time, O(1) space—our best solution. 2. Recursive Left-Rotation: O(n) time, O(n) space—an alternative.

Let’s dive into the best solution.

Best Solution: Iterative Left-Rotation Approach

Section link icon

Explanation

Iterative Left-Rotation transforms the tree by iterating from the root down the left chain, rotating each node:

  • For each node, save its left child, right child, and parent.
  • Make the left child the new root, set its right to the current node, and its left to the current node’s right.
  • Update pointers iteratively, avoiding recursion.

This is the best solution due to its O(n) time complexity, O(1) space usage (no recursion stack), and straightforward pointer adjustments, making it efficient and space-optimal.

For [1,2,3,4,5]:

  • Start at 1, rotate with 2.
  • Move to 2, rotate with 4.
  • Move to 4, rotate with 5.
  • Result: 5→4→2→3→1.

Step-by-Step Example

Assume a TreeNode class: class TreeNode: def __init__(self, val=0, left=None, right=None): self.val = val; self.left = left; self.right = right.

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

Goal: Return [5,4,2,null,null,3,1].

  • Start: curr = 1, prev = None, prev_right = None.
    • 1: left=2, right=3.
  • Step 1: curr = 1.
    • Save: left = 2, right = 3.
    • curr.left = prev_right → 1.left=None.
    • curr.right = prev → 1.right=None.
    • prev = currprev = 1.
    • prev_right = rightprev_right = 3.
    • curr = leftcurr = 2.
  • Step 2: curr = 2.
    • Save: left = 4, right = None.
    • 2.left=3, 2.right=1.
    • prev = 2, prev_right = None, curr = 4.
  • Step 3: curr = 4.
    • Save: left = 5, right = None.
    • 4.left=None, 4.right=2.
    • prev = 4, prev_right = None, curr = 5.
  • Step 4: curr = 5.
    • Save: left = None, right = None.
    • 5.left=None, 5.right=4.
    • prev = 5, curr = None.
  • Finish: Return prev = 5.

Example 2: root = []

Goal: Return null.

  • Start: curr = None, return None.
  • Finish: Return null.

Example 3: root = [1,2,null,3]

Goal: Return [3,2,null,1].

  • Start: curr = 1.
  • Step 1: curr = 1.
    • 1.left=2, 1.right=None.
    • 1.left=None, 1.right=None.
    • prev = 1, curr = 2.
  • Step 2: curr = 2.
    • 2.left=3, 2.right=None.
    • 2.left=None, 2.right=1.
    • prev = 2, curr = 3.
  • Step 3: curr = 3.
    • 3.left=None, 3.right=2.
    • prev = 3, curr = None.
  • Finish: Return 3.

How the Code Works (Iterative Left-Rotation) – Detailed Line-by-Line Explanation

Here’s the Python code with a thorough breakdown:

class TreeNode:
    def __init__(self, val=0, left=None, right=None):
        self.val = val
        self.left = left
        self.right = right

def upsideDownBinaryTree(root: TreeNode) -> TreeNode:
    # Line 1: Handle null or leaf case
    if not root or not root.left:
        # If empty or no left child (e.g., [] or leaf), return as is
        return root

    # Line 2: Initialize pointers
    curr = root
    # Current node (e.g., 1)
    prev = None
    # Previous parent (e.g., None)
    prev_right = None
    # Previous right child (e.g., None)

    # Line 3: Iterate down left chain
    while curr:
        # Continue until null (e.g., 1→2→4→5)

        # Line 4: Save current children
        left = curr.left
        # Left child (e.g., 2 for 1)
        right = curr.right
        # Right child (e.g., 3 for 1)

        # Line 5: Reassign pointers
        curr.left = prev_right
        # Left becomes prev right (e.g., 1.left=None)
        curr.right = prev
        # Right becomes prev (e.g., 1.right=None)

        # Line 6: Update pointers for next iteration
        prev = curr
        # Prev becomes current (e.g., 1)
        prev_right = right
        # Prev right becomes current right (e.g., 3)
        curr = left
        # Move to left child (e.g., 2)

    # Line 7: Return new root
    return prev
    # Last prev is new root (e.g., 5)

This detailed breakdown clarifies how iterative left-rotation efficiently flips the tree.

Alternative: Recursive Left-Rotation Approach

Section link icon

Explanation

Recursive Left-Rotation recursively processes the left chain:

  • Base case: Return the node if no left child.
  • Recurse on the left child, then reassign pointers to make it the new root.

It’s a practical alternative, also O(n) time, but uses O(n) space for the recursion stack, less space-efficient than iteration.

For [1,2,3,4,5]:

  • Recurse to 5, build up: 5→4→2→3→1.

Step-by-Step Example (Alternative)

For [1,2,3,4,5]:

  • Step 1: root = 1, recurse on 2.
  • Step 2: Recurse to 4, then 5 (base).
  • Step 3: Build: 5.right=4, 4.right=2, 2.left=3, 2.right=1.
  • Finish: Return 5.

How the Code Works (Recursive)

def upsideDownBinaryTreeRecursive(root: TreeNode) -> TreeNode:
    if not root or not root.left:
        return root
    new_root = upsideDownBinaryTreeRecursive(root.left)
    root.left.left = root.right
    root.left.right = root
    root.left = None
    root.right = None
    return new_root

Complexity

  • Iterative Left-Rotation:
    • Time: O(n) – single pass down left chain.
    • Space: O(1) – constant space.
  • Recursive Left-Rotation:
    • Time: O(n) – recursive traversal.
    • Space: O(n) – recursion stack.

Efficiency Notes

Iterative Left-Rotation is the best solution with O(n) time and O(1) space, offering efficiency and minimal memory—Recursive Left-Rotation matches time complexity but uses O(n) space due to recursion, making it elegant but less space-efficient.

Key Insights

  • Left Chain: Drives transformation.
  • Iteration: Space-efficient.
  • Recursion: Intuitive but costly.

Additional Example

Section link icon

For [2,3,null,4]:

  • Goal: [4,3,null,2].
  • Iterative: 4.right=3, 3.right=2.

Edge Cases

Section link icon
  • Empty Tree: []null.
  • Single Node: [1][1].
  • Right Leaf: [1,null,2][1,null,2].

Both solutions handle these well.

Final Thoughts

Section link icon

LeetCode 156: Binary Tree Upside Down in Python is a unique tree transformation challenge. The Iterative Left-Rotation solution excels with its space efficiency and clarity, while Recursive Left-Rotation offers a recursive approach. Want more? Try LeetCode 145: Binary Tree Postorder Traversal for tree basics or LeetCode 148: Sort List for list skills. Ready to practice? Solve LeetCode 156 on LeetCode with [1,2,3,4,5], aiming for [5,4,2,null,null,3,1]—test your skills now!