LeetCode 84: Largest Rectangle in Histogram Solution in Python Explained
Histograms might sound like a math class throwback, but LeetCode 84: Largest Rectangle in Histogram turns them into a fascinating coding challenge! This hard-level problem asks you to find the largest rectangle area in a histogram, where bar heights are given as an array. In this blog, we’ll crack it with Python, exploring two solutions—Stack-Based (our primary, efficient approach) and Brute Force (a simpler alternative). With step-by-step examples, detailed code breakdowns, and tips, you’ll conquer this problem. Let’s dive in!
Problem Statement
In LeetCode 84, you’re given an array heights
of non-negative integers, where each integer represents the height of a bar in a histogram, and the width of each bar is 1. Your task is to find the area of the largest rectangle that can be formed within the histogram. This builds on array manipulation skills, distinct from linked list problems like LeetCode 83: Remove Duplicates from Sorted List.
Constraints
- Array length: 1 <= heights.length <= 10^5
- Height values: 0 <= heights[i] <= 10^4
Example
Let’s see some cases:
Input: heights = [2,1,5,6,2,3]
Output: 10
Explanation: The largest rectangle is 5 and 6 (height 5, width 2), area = 5 * 2 = 10.
Input: heights = [2,4]
Output: 4
Explanation: Largest is height 2 over width 2 (2 * 2 = 4) or height 4 over width 1 (4 * 1 = 4).
Input: heights = [1,1]
Output: 2
Explanation: Height 1 over width 2 = 1 * 2 = 2.
These examples show we’re hunting for the biggest area possible.
Understanding the Problem
How do you solve LeetCode 84: Largest Rectangle in Histogram in Python? Take heights = [2,1,5,6,2,3]
. Each bar’s height is given, and we need the largest rectangle. A brute-force check of all possible rectangles is slow, so we’ll use a stack to track bars efficiently. This isn’t a search like LeetCode 1: Two Sum; it’s about geometry in arrays. We’ll explore:
1. Stack-Based: O(n) time, O(n) space—our best pick.
2. Brute Force: O(n²) time, O(1) space—a baseline.
Let’s start with the primary solution.
Solution 1: Stack-Based Approach (Primary)
Explanation
The stack-based method uses a stack to track bar indices in increasing height order. When we hit a smaller bar, we “pop” taller bars from the stack, calculating their rectangle areas using the current index as the right boundary and the stack’s next element as the left boundary. We add sentinel values (0s) at the start and end to simplify edge cases.
For heights = [2,1,5,6,2,3]
:
- Add bars to stack until a smaller one appears.
- Pop and calculate areas when a bar is smaller than the stack top.
- Final area: 10 (height 5 over width 2).
Step-by-Step Example
Example 1: heights = [2,1,5,6,2,3]
Goal: Return 10
.
- Start: heights = [0,2,1,5,6,2,3,0] (add sentinels), stack = [-1], max_area = 0.
- Step 1: i = 1, heights[1] = 2.
- Stack: [-1, 1].
- Step 2: i = 2, heights[2] = 1 < heights[1] = 2.
- Pop 1: Area = 2 * (2 - (-1) - 1) = 4, max_area = 4.
- Stack: [-1, 2].
- Step 3: i = 3, heights[3] = 5 > 1.
- Stack: [-1, 2, 3].
- Step 4: i = 4, heights[4] = 6 > 5.
- Stack: [-1, 2, 3, 4].
- Step 5: i = 5, heights[5] = 2 < 6.
- Pop 4: Area = 6 * (5 - 3 - 1) = 6, max_area = 6.
- Pop 3: Area = 5 * (5 - 2 - 1) = 10, max_area = 10.
- Stack: [-1, 2, 5].
- Step 6: i = 6, heights[6] = 3 > 2.
- Stack: [-1, 2, 5, 6].
- Step 7: i = 7, heights[7] = 0 < 3.
- Pop 6: Area = 3 * (7 - 5 - 1) = 3, max_area = 10.
- Pop 5: Area = 2 * (7 - 2 - 1) = 8, max_area = 10.
- Pop 2: Area = 1 * (7 - (-1) - 1) = 7, max_area = 10.
- Stack: [-1].
- Finish: Return max_area = 10.
Example 2: heights = [2,4]
Goal: Return 4
.
- Start: heights = [0,2,4,0], stack = [-1], max_area = 0.
- Step 1: i = 1, heights[1] = 2.
- Stack: [-1, 1].
- Step 2: i = 2, heights[2] = 4.
- Stack: [-1, 1, 2].
- Step 3: i = 3, heights[3] = 0 < 4.
- Pop 2: Area = 4 * (3 - 1 - 1) = 4, max_area = 4.
- Pop 1: Area = 2 * (3 - (-1) - 1) = 4, max_area = 4.
- Stack: [-1].
- Finish: Return 4.
Example 3: heights = [1,1]
Goal: Return 2
.
- Start: heights = [0,1,1,0], stack = [-1], max_area = 0.
- Step 1: i = 1, heights[1] = 1.
- Stack: [-1, 1].
- Step 2: i = 2, heights[2] = 1.
- Stack: [-1, 1, 2].
- Step 3: i = 3, heights[3] = 0 < 1.
- Pop 2: Area = 1 * (3 - 1 - 1) = 1, max_area = 1.
- Pop 1: Area = 1 * (3 - (-1) - 1) = 2, max_area = 2.
- Stack: [-1].
- Finish: Return 2.
How the Code Works (Stack-Based) – Detailed Line-by-Line Explanation
Here’s the Python code with a meticulous breakdown:
def largestRectangleArea(heights: list[int]) -> int:
# Line 1: Add sentinel values to simplify edge cases
heights = [0] + heights + [0]
# Prepends and appends 0 to the array (e.g., [2,1,5,6,2,3] becomes [0,2,1,5,6,2,3,0])
# Ensures all bars are processed by triggering pops at the start and end
# Line 2: Initialize stack with -1 as the left boundary
stack = [-1]
# Stack stores indices of bars in increasing height order
# -1 acts as a sentinel for the leftmost boundary, simplifying width calculations
# Line 3: Initialize max_area to track the largest rectangle
max_area = 0
# Starts at 0, will update whenever we find a larger rectangle area
# Line 4: Iterate through all indices of the modified heights array
for i in range(len(heights)):
# i goes from 0 to len(heights) - 1, covering all bars including sentinels
# e.g., for [0,2,1,5,6,2,3,0], i = 0 to 7
# Line 5: While stack has more than just -1 and current height is less than the top stack height
while stack[-1] != -1 and heights[stack[-1]] > heights[i]:
# Checks if stack isn’t just [-1] (empty of real bars) and if the bar at stack top is taller
# than the current bar (e.g., heights[stack[-1]] = 6 > heights[i] = 2)
# If true, we need to pop and calculate the area for the taller bar
# Line 6: Pop the top index from the stack
h = heights[stack.pop()]
# Removes the top index (e.g., 4 for height 6) and gets its height (h = 6)
# This bar can’t extend further right due to the smaller current bar
# Line 7: Calculate width using the current index and the new stack top
w = i - stack[-1] - 1
# Width is the difference between the current index (i) and the index below the popped one
# minus 1 (e.g., i = 5, stack[-1] = 3 after popping 4, w = 5 - 3 - 1 = 1)
# This is the span from the left boundary (stack[-1] + 1) to the right boundary (i - 1)
# Line 8: Calculate area and update max_area if larger
max_area = max(max_area, h * w)
# Area = height * width (e.g., 6 * 1 = 6), max_area updates if this is bigger
# Repeat this while loop if more bars need popping (e.g., next check 5 > 2)
# Line 9: Append the current index to the stack
stack.append(i)
# After processing pops, add the current index (e.g., 5 for height 2)
# Stack grows with indices of bars in increasing order until a smaller bar triggers pops
# Line 10: Return the largest area found
return max_area
# After the loop, max_area holds the largest rectangle area (e.g., 10)
This detailed breakdown clarifies each step, making the stack logic accessible.
Alternative: Brute Force Approach
Explanation
For each bar, calculate the maximum rectangle with that bar as the minimum height by expanding left and right until a smaller bar is hit. Check all bars to find the largest area.
For [2,1,5,6,2,3]
:
- Bar 2: Width 1, area = 2.
- Bar 1: Width 6, area = 6.
- Bar 5: Width 2, area = 10.
- Bar 6: Width 1, area = 6.
- Bar 2: Width 2, area = 4.
- Bar 3: Width 1, area = 3.
Max area: 10.
Step-by-Step Example (Alternative)
For [2,1,5,6,2,3]
:
- Bar 0 (2): Left to 0, right to 1, width = 1, area = 2.
- Bar 1 (1): Left to 0, right to 5, width = 6, area = 6.
- Bar 2 (5): Left to 1, right to 3, width = 2, area = 10.
- Bar 3 (6): Left to 2, right to 3, width = 1, area = 6.
- Bar 4 (2): Left to 1, right to 5, width = 2, area = 4.
- Bar 5 (3): Left to 4, right to 5, width = 1, area = 3.
- Finish: Max area = 10.
How the Code Works (Brute Force)
def largestRectangleAreaBrute(heights: list[int]) -> int:
max_area = 0
for i in range(len(heights)):
left = i
right = i
while left >= 0 and heights[left] >= heights[i]:
left -= 1
while right < len(heights) and heights[right] >= heights[i]:
right += 1
width = right - left - 1
max_area = max(max_area, heights[i] * width)
return max_area
Complexity
- Stack-Based:
- Time: O(n) – each bar pushed/popped once.
- Space: O(n) – stack size.
- Brute Force:
- Time: O(n²) – for each bar, scan left and right.
- Space: O(1) – no extra space.
Efficiency Notes
Stack-Based is the LeetCode-preferred solution for its O(n) efficiency, while Brute Force is a clear but slower baseline—useful for small arrays or learning.
Key Insights
- Stack Power: Tracks boundaries efficiently.
- Sentinels: Simplify edge cases.
- Height Limits: Area depends on the shortest bar.
Additional Example
For heights = [3,3,3]
:
- Goal: 9.
- Stack: Area = 3 * 3 = 9 (pops with sentinel).
Edge Cases
- Single Bar: [5] → 5.
- All Same: [2,2,2] → 6.
- Zeros: [0,0] → 0.
Both solutions handle these well.
Final Thoughts
LeetCode 84: Largest Rectangle in Histogram in Python is a brilliant test of stack skills. The Stack-Based solution is fast and clever, while Brute Force offers a straightforward start. Want more? Try LeetCode 85: Maximal Rectangle for a 2D twist or LeetCode 80: Remove Duplicates from Sorted Array II for array fun. Ready to practice? Solve LeetCode 84 on LeetCode with [2,1,5,6,2,3]
and aim for 10
—test your skills now!
- Array length: 1 <= heights.length <= 10^5
- Height values: 0 <= heights[i] <= 10^4
Example
Let’s see some cases:
Input: heights = [2,1,5,6,2,3]
Output: 10
Explanation: The largest rectangle is 5 and 6 (height 5, width 2), area = 5 * 2 = 10.
Input: heights = [2,4]
Output: 4
Explanation: Largest is height 2 over width 2 (2 * 2 = 4) or height 4 over width 1 (4 * 1 = 4).
Input: heights = [1,1]
Output: 2
Explanation: Height 1 over width 2 = 1 * 2 = 2.
These examples show we’re hunting for the biggest area possible.
Understanding the Problem
How do you solve LeetCode 84: Largest Rectangle in Histogram in Python? Take heights = [2,1,5,6,2,3]
. Each bar’s height is given, and we need the largest rectangle. A brute-force check of all possible rectangles is slow, so we’ll use a stack to track bars efficiently. This isn’t a search like LeetCode 1: Two Sum; it’s about geometry in arrays. We’ll explore:
1. Stack-Based: O(n) time, O(n) space—our best pick.
2. Brute Force: O(n²) time, O(1) space—a baseline.
Let’s start with the primary solution.
Solution 1: Stack-Based Approach (Primary)
Explanation
The stack-based method uses a stack to track bar indices in increasing height order. When we hit a smaller bar, we “pop” taller bars from the stack, calculating their rectangle areas using the current index as the right boundary and the stack’s next element as the left boundary. We add sentinel values (0s) at the start and end to simplify edge cases.
For heights = [2,1,5,6,2,3]
:
- Add bars to stack until a smaller one appears.
- Pop and calculate areas when a bar is smaller than the stack top.
- Final area: 10 (height 5 over width 2).
Step-by-Step Example
Example 1: heights = [2,1,5,6,2,3]
Goal: Return 10
.
- Start: heights = [0,2,1,5,6,2,3,0] (add sentinels), stack = [-1], max_area = 0.
- Step 1: i = 1, heights[1] = 2.
- Stack: [-1, 1].
- Step 2: i = 2, heights[2] = 1 < heights[1] = 2.
- Pop 1: Area = 2 * (2 - (-1) - 1) = 4, max_area = 4.
- Stack: [-1, 2].
- Step 3: i = 3, heights[3] = 5 > 1.
- Stack: [-1, 2, 3].
- Step 4: i = 4, heights[4] = 6 > 5.
- Stack: [-1, 2, 3, 4].
- Step 5: i = 5, heights[5] = 2 < 6.
- Pop 4: Area = 6 * (5 - 3 - 1) = 6, max_area = 6.
- Pop 3: Area = 5 * (5 - 2 - 1) = 10, max_area = 10.
- Stack: [-1, 2, 5].
- Step 6: i = 6, heights[6] = 3 > 2.
- Stack: [-1, 2, 5, 6].
- Step 7: i = 7, heights[7] = 0 < 3.
- Pop 6: Area = 3 * (7 - 5 - 1) = 3, max_area = 10.
- Pop 5: Area = 2 * (7 - 2 - 1) = 8, max_area = 10.
- Pop 2: Area = 1 * (7 - (-1) - 1) = 7, max_area = 10.
- Stack: [-1].
- Finish: Return max_area = 10.
Example 2: heights = [2,4]
Goal: Return 4
.
- Start: heights = [0,2,4,0], stack = [-1], max_area = 0.
- Step 1: i = 1, heights[1] = 2.
- Stack: [-1, 1].
- Step 2: i = 2, heights[2] = 4.
- Stack: [-1, 1, 2].
- Step 3: i = 3, heights[3] = 0 < 4.
- Pop 2: Area = 4 * (3 - 1 - 1) = 4, max_area = 4.
- Pop 1: Area = 2 * (3 - (-1) - 1) = 4, max_area = 4.
- Stack: [-1].
- Finish: Return 4.
Example 3: heights = [1,1]
Goal: Return 2
.
- Start: heights = [0,1,1,0], stack = [-1], max_area = 0.
- Step 1: i = 1, heights[1] = 1.
- Stack: [-1, 1].
- Step 2: i = 2, heights[2] = 1.
- Stack: [-1, 1, 2].
- Step 3: i = 3, heights[3] = 0 < 1.
- Pop 2: Area = 1 * (3 - 1 - 1) = 1, max_area = 1.
- Pop 1: Area = 1 * (3 - (-1) - 1) = 2, max_area = 2.
- Stack: [-1].
- Finish: Return 2.
How the Code Works (Stack-Based) – Detailed Line-by-Line Explanation
Here’s the Python code with a meticulous breakdown:
def largestRectangleArea(heights: list[int]) -> int:
# Line 1: Add sentinel values to simplify edge cases
heights = [0] + heights + [0]
# Prepends and appends 0 to the array (e.g., [2,1,5,6,2,3] becomes [0,2,1,5,6,2,3,0])
# Ensures all bars are processed by triggering pops at the start and end
# Line 2: Initialize stack with -1 as the left boundary
stack = [-1]
# Stack stores indices of bars in increasing height order
# -1 acts as a sentinel for the leftmost boundary, simplifying width calculations
# Line 3: Initialize max_area to track the largest rectangle
max_area = 0
# Starts at 0, will update whenever we find a larger rectangle area
# Line 4: Iterate through all indices of the modified heights array
for i in range(len(heights)):
# i goes from 0 to len(heights) - 1, covering all bars including sentinels
# e.g., for [0,2,1,5,6,2,3,0], i = 0 to 7
# Line 5: While stack has more than just -1 and current height is less than the top stack height
while stack[-1] != -1 and heights[stack[-1]] > heights[i]:
# Checks if stack isn’t just [-1] (empty of real bars) and if the bar at stack top is taller
# than the current bar (e.g., heights[stack[-1]] = 6 > heights[i] = 2)
# If true, we need to pop and calculate the area for the taller bar
# Line 6: Pop the top index from the stack
h = heights[stack.pop()]
# Removes the top index (e.g., 4 for height 6) and gets its height (h = 6)
# This bar can’t extend further right due to the smaller current bar
# Line 7: Calculate width using the current index and the new stack top
w = i - stack[-1] - 1
# Width is the difference between the current index (i) and the index below the popped one
# minus 1 (e.g., i = 5, stack[-1] = 3 after popping 4, w = 5 - 3 - 1 = 1)
# This is the span from the left boundary (stack[-1] + 1) to the right boundary (i - 1)
# Line 8: Calculate area and update max_area if larger
max_area = max(max_area, h * w)
# Area = height * width (e.g., 6 * 1 = 6), max_area updates if this is bigger
# Repeat this while loop if more bars need popping (e.g., next check 5 > 2)
# Line 9: Append the current index to the stack
stack.append(i)
# After processing pops, add the current index (e.g., 5 for height 2)
# Stack grows with indices of bars in increasing order until a smaller bar triggers pops
# Line 10: Return the largest area found
return max_area
# After the loop, max_area holds the largest rectangle area (e.g., 10)
This detailed breakdown clarifies each step, making the stack logic accessible.
Alternative: Brute Force Approach
Explanation
For each bar, calculate the maximum rectangle with that bar as the minimum height by expanding left and right until a smaller bar is hit. Check all bars to find the largest area.
For [2,1,5,6,2,3]
:
- Bar 2: Width 1, area = 2.
- Bar 1: Width 6, area = 6.
- Bar 5: Width 2, area = 10.
- Bar 6: Width 1, area = 6.
- Bar 2: Width 2, area = 4.
- Bar 3: Width 1, area = 3.
Max area: 10.
Step-by-Step Example (Alternative)
For [2,1,5,6,2,3]
:
- Bar 0 (2): Left to 0, right to 1, width = 1, area = 2.
- Bar 1 (1): Left to 0, right to 5, width = 6, area = 6.
- Bar 2 (5): Left to 1, right to 3, width = 2, area = 10.
- Bar 3 (6): Left to 2, right to 3, width = 1, area = 6.
- Bar 4 (2): Left to 1, right to 5, width = 2, area = 4.
- Bar 5 (3): Left to 4, right to 5, width = 1, area = 3.
- Finish: Max area = 10.
How the Code Works (Brute Force)
def largestRectangleAreaBrute(heights: list[int]) -> int:
max_area = 0
for i in range(len(heights)):
left = i
right = i
while left >= 0 and heights[left] >= heights[i]:
left -= 1
while right < len(heights) and heights[right] >= heights[i]:
right += 1
width = right - left - 1
max_area = max(max_area, heights[i] * width)
return max_area
Complexity
- Stack-Based:
- Time: O(n) – each bar pushed/popped once.
- Space: O(n) – stack size.
- Brute Force:
- Time: O(n²) – for each bar, scan left and right.
- Space: O(1) – no extra space.
Efficiency Notes
Stack-Based is the LeetCode-preferred solution for its O(n) efficiency, while Brute Force is a clear but slower baseline—useful for small arrays or learning.
Key Insights
- Stack Power: Tracks boundaries efficiently.
- Sentinels: Simplify edge cases.
- Height Limits: Area depends on the shortest bar.
Additional Example
For heights = [3,3,3]
:
- Goal: 9.
- Stack: Area = 3 * 3 = 9 (pops with sentinel).
Edge Cases
- Single Bar: [5] → 5.
- All Same: [2,2,2] → 6.
- Zeros: [0,0] → 0.
Both solutions handle these well.
Final Thoughts
LeetCode 84: Largest Rectangle in Histogram in Python is a brilliant test of stack skills. The Stack-Based solution is fast and clever, while Brute Force offers a straightforward start. Want more? Try LeetCode 85: Maximal Rectangle for a 2D twist or LeetCode 80: Remove Duplicates from Sorted Array II for array fun. Ready to practice? Solve LeetCode 84 on LeetCode with [2,1,5,6,2,3]
and aim for 10
—test your skills now!
URL, Title, and Description
URL
/leetcode-84-largest-rectangle-in-histogram
Title
LeetCode 84: Largest Rectangle in Histogram Solution in Python Explained
Description
Master LeetCode 84: Largest Rectangle in Histogram with Python! Step-by-step examples and code included.
---Below is a detailed blog post for LeetCode 84: Largest Rectangle in Histogram, exceeding 2000 words, optimized for SEO, and beginner-friendly. It follows your template, includes internal linking with the existing URL pattern (/leetcode/[problem-number]/[problem-name]
), a Python basics link, step-by-step examples, two solutions (Stack-Based as primary, Brute Force as alternative), and an external CTA in "Final Thoughts." The URL for the "URL, Title, and Description" section uses the modified pattern (/leetcode-84-largest-rectangle-in-histogram
). Per your request, I’ve provided a detailed line-by-line explanation for the primary solution (Stack-Based). I’ve stopped after the URL, title, and description section.
LeetCode 84: Largest Rectangle in Histogram Solution in Python Explained
Histograms might sound like a math class throwback, but LeetCode 84: Largest Rectangle in Histogram turns them into a fascinating coding challenge! This hard-level problem asks you to find the largest rectangle area in a histogram, where bar heights are given as an array. In this blog, we’ll crack it with Python, exploring two solutions—Stack-Based (our primary, efficient approach) and Brute Force (a simpler alternative). With step-by-step examples, detailed code breakdowns, and tips, you’ll conquer this problem. Let’s dive in!
Problem Statement
In LeetCode 84, you’re given an array heights
of non-negative integers, where each integer represents the height of a bar in a histogram, and the width of each bar is 1. Your task is to find the area of the largest rectangle that can be formed within the histogram. This builds on array manipulation skills, distinct from linked list problems like LeetCode 83: Remove Duplicates from Sorted List.
Constraints
- Array length: 1 <= heights.length <= 10^5
- Height values: 0 <= heights[i] <= 10^4
Example
Let’s see some cases:
Input: heights = [2,1,5,6,2,3]
Output: 10
Explanation: The largest rectangle is 5 and 6 (height 5, width 2), area = 5 * 2 = 10.
Input: heights = [2,4]
Output: 4
Explanation: Largest is height 2 over width 2 (2 * 2 = 4) or height 4 over width 1 (4 * 1 = 4).
Input: heights = [1,1]
Output: 2
Explanation: Height 1 over width 2 = 1 * 2 = 2.
These examples show we’re hunting for the biggest area possible.
Understanding the Problem
How do you solve LeetCode 84: Largest Rectangle in Histogram in Python? Take heights = [2,1,5,6,2,3]
. Each bar’s height is given, and we need the largest rectangle. A brute-force check of all possible rectangles is slow, so we’ll use a stack to track bars efficiently. This isn’t a search like LeetCode 1: Two Sum; it’s about geometry in arrays. We’ll explore:
1. Stack-Based: O(n) time, O(n) space—our best pick.
2. Brute Force: O(n²) time, O(1) space—a baseline.
Let’s start with the primary solution.
Solution 1: Stack-Based Approach (Primary)
Explanation
The stack-based method uses a stack to track bar indices in increasing height order. When we hit a smaller bar, we “pop” taller bars from the stack, calculating their rectangle areas using the current index as the right boundary and the stack’s next element as the left boundary. We add sentinel values (0s) at the start and end to simplify edge cases.
For heights = [2,1,5,6,2,3]
:
- Add bars to stack until a smaller one appears.
- Pop and calculate areas when a bar is smaller than the stack top.
- Final area: 10 (height 5 over width 2).
Step-by-Step Example
Example 1: heights = [2,1,5,6,2,3]
Goal: Return 10
.
- Start: heights = [0,2,1,5,6,2,3,0] (add sentinels), stack = [-1], max_area = 0.
- Step 1: i = 1, heights[1] = 2.
- Stack: [-1, 1].
- Step 2: i = 2, heights[2] = 1 < heights[1] = 2.
- Pop 1: Area = 2 * (2 - (-1) - 1) = 4, max_area = 4.
- Stack: [-1, 2].
- Step 3: i = 3, heights[3] = 5 > 1.
- Stack: [-1, 2, 3].
- Step 4: i = 4, heights[4] = 6 > 5.
- Stack: [-1, 2, 3, 4].
- Step 5: i = 5, heights[5] = 2 < 6.
- Pop 4: Area = 6 * (5 - 3 - 1) = 6, max_area = 6.
- Pop 3: Area = 5 * (5 - 2 - 1) = 10, max_area = 10.
- Stack: [-1, 2, 5].
- Step 6: i = 6, heights[6] = 3 > 2.
- Stack: [-1, 2, 5, 6].
- Step 7: i = 7, heights[7] = 0 < 3.
- Pop 6: Area = 3 * (7 - 5 - 1) = 3, max_area = 10.
- Pop 5: Area = 2 * (7 - 2 - 1) = 8, max_area = 10.
- Pop 2: Area = 1 * (7 - (-1) - 1) = 7, max_area = 10.
- Stack: [-1].
- Finish: Return max_area = 10.
Example 2: heights = [2,4]
Goal: Return 4
.
- Start: heights = [0,2,4,0], stack = [-1], max_area = 0.
- Step 1: i = 1, heights[1] = 2.
- Stack: [-1, 1].
- Step 2: i = 2, heights[2] = 4.
- Stack: [-1, 1, 2].
- Step 3: i = 3, heights[3] = 0 < 4.
- Pop 2: Area = 4 * (3 - 1 - 1) = 4, max_area = 4.
- Pop 1: Area = 2 * (3 - (-1) - 1) = 4, max_area = 4.
- Stack: [-1].
- Finish: Return 4.
Example 3: heights = [1,1]
Goal: Return 2
.
- Start: heights = [0,1,1,0], stack = [-1], max_area = 0.
- Step 1: i = 1, heights[1] = 1.
- Stack: [-1, 1].
- Step 2: i = 2, heights[2] = 1.
- Stack: [-1, 1, 2].
- Step 3: i = 3, heights[3] = 0 < 1.
- Pop 2: Area = 1 * (3 - 1 - 1) = 1, max_area = 1.
- Pop 1: Area = 1 * (3 - (-1) - 1) = 2, max_area = 2.
- Stack: [-1].
- Finish: Return 2.
How the Code Works (Stack-Based) – Detailed Line-by-Line Explanation
Here’s the Python code with a meticulous breakdown:
def largestRectangleArea(heights: list[int]) -> int:
# Line 1: Add sentinel values to simplify edge cases
heights = [0] + heights + [0]
# Prepends and appends 0 to the array (e.g., [2,1,5,6,2,3] becomes [0,2,1,5,6,2,3,0])
# Ensures all bars are processed by triggering pops at the start and end
# Line 2: Initialize stack with -1 as the left boundary
stack = [-1]
# Stack stores indices of bars in increasing height order
# -1 acts as a sentinel for the leftmost boundary, simplifying width calculations
# Line 3: Initialize max_area to track the largest rectangle
max_area = 0
# Starts at 0, will update whenever we find a larger rectangle area
# Line 4: Iterate through all indices of the modified heights array
for i in range(len(heights)):
# i goes from 0 to len(heights) - 1, covering all bars including sentinels
# e.g., for [0,2,1,5,6,2,3,0], i = 0 to 7
# Line 5: While stack has more than just -1 and current height is less than the top stack height
while stack[-1] != -1 and heights[stack[-1]] > heights[i]:
# Checks if stack isn’t just [-1] (empty of real bars) and if the bar at stack top is taller
# than the current bar (e.g., heights[stack[-1]] = 6 > heights[i] = 2)
# If true, we need to pop and calculate the area for the taller bar
# Line 6: Pop the top index from the stack
h = heights[stack.pop()]
# Removes the top index (e.g., 4 for height 6) and gets its height (h = 6)
# This bar can’t extend further right due to the smaller current bar
# Line 7: Calculate width using the current index and the new stack top
w = i - stack[-1] - 1
# Width is the difference between the current index (i) and the index below the popped one
# minus 1 (e.g., i = 5, stack[-1] = 3 after popping 4, w = 5 - 3 - 1 = 1)
# This is the span from the left boundary (stack[-1] + 1) to the right boundary (i - 1)
# Line 8: Calculate area and update max_area if larger
max_area = max(max_area, h * w)
# Area = height * width (e.g., 6 * 1 = 6), max_area updates if this is bigger
# Repeat this while loop if more bars need popping (e.g., next check 5 > 2)
# Line 9: Append the current index to the stack
stack.append(i)
# After processing pops, add the current index (e.g., 5 for height 2)
# Stack grows with indices of bars in increasing order until a smaller bar triggers pops
# Line 10: Return the largest area found
return max_area
# After the loop, max_area holds the largest rectangle area (e.g., 10)
This detailed breakdown clarifies each step, making the stack logic accessible.
Alternative: Brute Force Approach
Explanation
For each bar, calculate the maximum rectangle with that bar as the minimum height by expanding left and right until a smaller bar is hit. Check all bars to find the largest area.
For [2,1,5,6,2,3]
:
- Bar 2: Width 1, area = 2.
- Bar 1: Width 6, area = 6.
- Bar 5: Width 2, area = 10.
- Bar 6: Width 1, area = 6.
- Bar 2: Width 2, area = 4.
- Bar 3: Width 1, area = 3.
Max area: 10.
Step-by-Step Example (Alternative)
For [2,1,5,6,2,3]
:
- Bar 0 (2): Left to 0, right to 1, width = 1, area = 2.
- Bar 1 (1): Left to 0, right to 5, width = 6, area = 6.
- Bar 2 (5): Left to 1, right to 3, width = 2, area = 10.
- Bar 3 (6): Left to 2, right to 3, width = 1, area = 6.
- Bar 4 (2): Left to 1, right to 5, width = 2, area = 4.
- Bar 5 (3): Left to 4, right to 5, width = 1, area = 3.
- Finish: Max area = 10.
How the Code Works (Brute Force)
def largestRectangleAreaBrute(heights: list[int]) -> int:
max_area = 0
for i in range(len(heights)):
left = i
right = i
while left >= 0 and heights[left] >= heights[i]:
left -= 1
while right < len(heights) and heights[right] >= heights[i]:
right += 1
width = right - left - 1
max_area = max(max_area, heights[i] * width)
return max_area
Complexity
- Stack-Based:
- Time: O(n) – each bar pushed/popped once.
- Space: O(n) – stack size.
- Brute Force:
- Time: O(n²) – for each bar, scan left and right.
- Space: O(1) – no extra space.
Efficiency Notes
Stack-Based is the LeetCode-preferred solution for its O(n) efficiency, while Brute Force is a clear but slower baseline—useful for small arrays or learning.
Key Insights
- Stack Power: Tracks boundaries efficiently.
- Sentinels: Simplify edge cases.
- Height Limits: Area depends on the shortest bar.
Additional Example
For heights = [3,3,3]
:
- Goal: 9.
- Stack: Area = 3 * 3 = 9 (pops with sentinel).
Edge Cases
- Single Bar: [5] → 5.
- All Same: [2,2,2] → 6.
- Zeros: [0,0] → 0.
Both solutions handle these well.
Final Thoughts
LeetCode 84: Largest Rectangle in Histogram in Python is a brilliant test of stack skills. The Stack-Based solution is fast and clever, while Brute Force offers a straightforward start. Want more? Try LeetCode 85: Maximal Rectangle for a 2D twist or LeetCode 80: Remove Duplicates from Sorted Array II for array fun. Ready to practice? Solve LeetCode 84 on LeetCode with [2,1,5,6,2,3]
and aim for 10
—test your skills now!
URL, Title, and Description
URL
/leetcode-84-largest-rectangle-in-histogram
Title
LeetCode 84: Largest Rectangle in Histogram Solution in Python Explained
Description
Master LeetCode 84: Largest Rectangle in Histogram with Python! Step-by-step examples and code included.