LeetCode 264: Ugly Number II Solution in Python – A Step-by-Step Guide

Imagine you’re on a treasure hunt to find special numbers—called "ugly numbers"—that are built only from the factors 2, 3, and 5, and you need to track down the nth one in order, like the 10th in a growing list. That’s the exciting challenge of LeetCode 264: Ugly Number II! This medium-level problem asks you to find the nth ugly number, where ugly numbers are those whose prime factors are limited to 2, 3, or 5 (like 1, 2, 3, 4, 5, 6, 8, 9, 10, 12). Using Python, we’ll explore two solutions: the Best Solution, a dynamic programming approach with multiple pointers, and an Alternative Solution, a min-heap method. With detailed examples, clear code, and friendly, easy-to-follow explanations—especially for the best solution—this guide will help you uncover ugly numbers and boost your coding skills. Let’s start hunting for that nth ugly number!

What Is LeetCode 264: Ugly Number II?

Section link icon

In LeetCode 264: Ugly Number II, you’re given an integer n, and your task is to return the nth ugly number in the sequence of ugly numbers, where an ugly number has only 2, 3, or 5 as its prime factors. The sequence starts with 1 (no factors), then 2, 3, 4, 5, 6, 8, 9, 10, 12, and so on. This problem builds on LeetCode 263: Ugly Number, moving from checking a single number to generating a sequence up to the nth term.

Problem Statement

  • Input: An integer n.
  • Output: An integer—the nth ugly number in the sequence.
  • Rules: Ugly numbers have prime factors only of 2, 3, or 5; sequence starts at 1; order matters.

Constraints

  • n: 1 to 1690 (due to integer limits and sequence growth).

Examples

  • Input: n = 10
    • Output: 12 (sequence: 1, 2, 3, 4, 5, 6, 8, 9, 10, 12).
  • Input: n = 1
    • Output: 1 (first ugly number).
  • Input: n = 7
    • Output: 8 (sequence: 1, 2, 3, 4, 5, 6, 8).

Understanding the Problem: Finding the nth Ugly Number

Section link icon

To solve LeetCode 264: Ugly Number II in Python, we need to generate ugly numbers in order—starting with 1—and keep going until we reach the nth one. Ugly numbers are products of powers of 2, 3, and 5 (e.g., 1 = 2^0 * 3^0 * 5^0, 6 = 2^1 * 3^1 * 5^0). A simple way—checking every number to see if it’s ugly—works but is way too slow. We’ll use two methods: 1. Best Solution (Dynamic Programming with Multiple Pointers): O(n) time—fast and clever. 2. Alternative Solution (Min-Heap): O(n log n)—steady but uses more memory.

Let’s dive into the best solution with a friendly, detailed walkthrough.

Best Solution: Dynamic Programming with Multiple Pointers

Section link icon

Why This Is the Best Solution

The dynamic programming approach with multiple pointers is the top pick for LeetCode 264 because it runs in O(n) time—super quick—and uses O(n) space to build the sequence directly, avoiding unnecessary checks. It’s like having three helpers pointing to the next numbers to multiply by 2, 3, and 5, picking the smallest each time, which keeps everything smooth and efficient.

How It Works

Picture this solution as running a number factory: you start with 1, then use three assembly lines—one for 2s, one for 3s, and one for 5s—to build new ugly numbers by multiplying existing ones. Each line has a pointer tracking which number it’s working on, and you always pick the smallest result to add next. Here’s how it works, step-by-step, explained simply:

  • Step 1: Start with a List:
    • Make a list ugly to hold the sequence, starting with [1].
  • Step 2: Set Up Pointers:
    • Use three pointers: p2, p3, p5, all starting at 0 (pointing to the first ugly number, 1).
    • These track which ugly number to multiply by 2, 3, or 5 next.
  • Step 3: Build the Sequence:
    • For each step up to n:
      • Calculate next candidates: ugly[p2] * 2, ugly[p3] * 3, ugly[p5] * 5.
      • Pick the smallest one (call it next_ugly).
      • Add it to the list.
      • Move the pointer(s) forward for whichever factor(s) produced it.
  • Step 4: Avoid Duplicates:
    • If next_ugly comes from multiple factors (e.g., 6 = 2 * 3 = 3 * 2), move all matching pointers.
  • Step 5: Get the nth Number:
    • The list grows to size n; return the last one (ugly[n-1]).

It’s like a race where 2, 3, and 5 compete to make the next smallest number—pick the winner and keep going!

Step-by-Step Example

Example: n = 10

  • Step 1: ugly = [1], p2 = 0, p3 = 0, p5 = 0.
  • Step 2: Build up to 10:
    • Next: min(1*2, 1*3, 1*5) = 2, ugly = [1, 2], p2 = 1.
    • Next: min(2*2, 1*3, 1*5) = 3, ugly = [1, 2, 3], p3 = 1.
    • Next: min(2*2, 3*3, 1*5) = 4, ugly = [1, 2, 3, 4], p2 = 2.
    • Next: min(3*2, 3*3, 1*5) = 5, ugly = [1, 2, 3, 4, 5], p5 = 1.
    • Next: min(3*2, 3*3, 5*5) = 6, ugly = [1, 2, 3, 4, 5, 6], p2 = 3.
    • Next: min(4*2, 3*3, 5*5) = 8, ugly = [1, 2, 3, 4, 5, 6, 8], p2 = 4.
    • Next: min(5*2, 3*3, 5*5) = 9, ugly = [1, 2, 3, 4, 5, 6, 8, 9], p3 = 2.
    • Next: min(5*2, 9*3, 5*5) = 10, ugly = [1, 2, 3, 4, 5, 6, 8, 9, 10], p2 = 5.
    • Next: min(6*2, 9*3, 5*5) = 12, ugly = [1, 2, 3, 4, 5, 6, 8, 9, 10, 12], p2 = 6.
  • Step 3: n = 10, return ugly[9] = 12.
  • Result: 12.

Code with Detailed Line-by-Line Explanation

Here’s the Python code, explained in a friendly way:

class Solution:
    def nthUglyNumber(self, n: int) -> int:
        # Step 1: Set up the ugly number list
        ugly = [1] * n  # Make room for n numbers, start with 1

        # Step 2: Initialize pointers and factors
        p2 = p3 = p5 = 0  # Start all pointers at index 0
        next2, next3, next5 = 2, 3, 5  # First multiples of 1

        # Step 3: Build the sequence
        for i in range(1, n):
            # Find the smallest next ugly number
            next_ugly = min(next2, next3, next5)
            ugly[i] = next_ugly

            # Step 4: Update pointers for used factors
            if next_ugly == next2:
                p2 += 1
                next2 = ugly[p2] * 2
            if next_ugly == next3:
                p3 += 1
                next3 = ugly[p3] * 3
            if next_ugly == next5:
                p5 += 1
                next5 = ugly[p5] * 5

        # Step 5: Return the nth ugly number
        return ugly[n-1]
  • Line 3-4: Create a list of size n, starting with 1.
  • Line 6-7: Set pointers to 0 and initial next values (2, 3, 5).
  • Line 9-19: Loop n-1 times to fill the list:
    • Pick the smallest of next2, next3, next5.
    • Add it to ugly.
    • Move the pointer(s) forward and update the next value(s) for any factor used.
  • Line 21: Return the last (nth) number.
  • Time Complexity: O(n)—builds n numbers linearly.
  • Space Complexity: O(n)—stores the sequence.

This solution is like a well-oiled machine—three pointers churn out ugly numbers in perfect order!

Alternative Solution: Min-Heap Approach

Section link icon

Why an Alternative Approach?

The min-heap method uses a priority queue to keep track of the next smallest ugly number, like a conveyor belt spitting out numbers in order. It’s a bit slower (O(n log n)) and uses more space, but it’s a fun way to see how a heap can organize things for you.

How It Works

Imagine you’re managing a line of numbers waiting to join the ugly list: you start with 1, multiply it by 2, 3, and 5, and keep picking the smallest one to add next, tossing its multiples into the line. Here’s how it works, step-by-step:

  • Step 1: Set Up a Heap:
    • Use a min-heap to store potential ugly numbers, starting with 1.
  • Step 2: Build the Sequence:
    • Pop the smallest number, add it to your list.
    • Push its multiples (times 2, 3, 5) into the heap.
  • Step 3: Avoid Duplicates:
    • Use a set to skip numbers already added (e.g., 6 from 2*3 and 3*2).
  • Step 4: Repeat n Times:
    • Keep going until you’ve got n numbers.

It’s like a number buffet—grab the smallest dish and refill the tray with its cousins!

Step-by-Step Example

Example: n = 5

  • Step 1: Heap = [1], Seen = {1}, Result = [].
  • Step 2:
    • Pop 1, Result = [1], Add 2, 3, 5 → Heap = [2, 3, 5], Seen = {1, 2, 3, 5}.
    • Pop 2, Result = [1, 2], Add 4, 6, 10 → Heap = [3, 5, 4, 6, 10], Seen adds 4, 6, 10.
    • Pop 3, Result = [1, 2, 3], Add 6 (skip), 9, 15 → Heap = [4, 5, 6, 10, 9, 15], Seen adds 9, 15.
    • Pop 4, Result = [1, 2, 3, 4], Add 8, 12, 20 → Heap = [5, 6, 9, 10, 15, 8, 12, 20], Seen adds 8, 12, 20.
    • Pop 5, Result = [1, 2, 3, 4, 5], Done.
  • Result: 5.

Code for Min-Heap Approach

import heapq

class Solution:
    def nthUglyNumber(self, n: int) -> int:
        # Step 1: Set up heap and seen set
        heap = [1]
        seen = {1}
        factors = [2, 3, 5]

        # Step 2: Get n ugly numbers
        for _ in range(n - 1):
            curr = heapq.heappop(heap)
            for factor in factors:
                next_num = curr * factor
                if next_num not in seen:
                    heapq.heappush(heap, next_num)
                    seen.add(next_num)

        # Step 3: Return the nth number
        return heapq.heappop(heap)
  • Time Complexity: O(n log n)—heap operations for n numbers.
  • Space Complexity: O(n)—heap and set size.

It’s a steady builder but heavier on resources.

Comparing the Two Solutions

Section link icon
  • Best Solution (DP with Pointers):
    • Pros: O(n) time, O(n) space, fast and lean.
    • Cons: Pointer logic might take a moment to click.
  • Alternative Solution (Min-Heap):
    • Pros: O(n log n) time, heap makes order clear.
    • Cons: O(n) space, slower with heap ops.

DP wins for efficiency.

Additional Examples and Edge Cases

Section link icon

First Few

  • n = 11
  • n = 33

Larger n

  • n = 1524 (sequence up to 24)

Both handle these smoothly.

Complexity Breakdown

Section link icon
  • DP with Pointers:
    • Time: O(n)—linear build.
    • Space: O(n)—list size.
  • Min-Heap:
    • Time: O(n log n)—heap operations.
    • Space: O(n)—heap and set.

DP is the speed king.

Key Takeaways

Section link icon
  • DP Pointers: Build in order with helpers.
  • Min-Heap: Sort with a queue.
  • Ugly Numbers: Grow from 2, 3, 5.
  • Python Tip: Lists and heaps are versatile—see [Python Basics](/python/basics).

Final Thoughts: Hunt Ugly Numbers Like a Pro

Section link icon

LeetCode 264: Ugly Number II in Python is a thrilling number chase. The DP solution races ahead with pointers, while the min-heap offers a steady climb. Want more? Try LeetCode 263: Ugly Number or LeetCode 313: Super Ugly Number. Ready to hunt? Head to Solve LeetCode 264 on LeetCode and nab that nth ugly number today!