694. Number of Distinct Islands - Detailed Explanation
Problem Statement
Description:
You are given a 2D grid of integers (0’s and 1’s) where:
- 1 represents land.
- 0 represents water.
An island is a group of connected 1’s (land cells) connected horizontally or vertically. Two islands are considered the same if and only if one island can be translated (moved horizontally and vertically) to match the other exactly. Your goal is to determine how many distinct islands exist in the grid.
Example 1:
- Input:
grid = [ [1,1,0,0,0], [1,0,0,0,0], [0,0,0,1,1], [0,0,0,1,1] ]
- Output:
1
- Explanation:
There are two islands in the grid. One island is at the top-left and the other at the bottom-right. Although they are in different positions, their shapes are identical if you translate the top-left island to the bottom-right (both form an “L” shape).
Example 2:
- Input:
grid = [ [1,1,0,1,1], [1,0,0,0,0], [0,0,0,0,1], [1,1,0,1,1] ]
- Output:
3
- Explanation:
The grid contains multiple islands with different shapes. After translating each island to a canonical position (using relative coordinates), you will find three distinct shapes.
Constraints:
- The dimensions of the grid can be up to around 50×50 (or similar constraints).
- The grid contains only 0’s and 1’s.
- Two islands are considered the same if one can be translated to match the other exactly.
Hints
-
Hint 1:
Think about how you can capture the shape of an island regardless of its position. A common technique is to record the relative positions of each cell in the island compared to a starting point (for example, the top-left cell of that island). -
Hint 2:
Use a Depth-First Search (DFS) or Breadth-First Search (BFS) to traverse each island. As you explore, record each cell’s relative coordinate (e.g., (row - start_row, col - start_col)). -
Hint 3:
Once you have the relative positions for an island, you can convert the list of coordinates into a tuple (or another immutable data structure) and add it to a set. The size of the set will then give you the number of distinct island shapes.
Approach 1: Depth-First Search (DFS) with Relative Coordinates
Idea
-
Traverse the Grid:
Iterate over every cell in the grid. When you find a cell with value 1 (land) that has not been visited, start a DFS from that cell. -
Record Relative Positions:
For the DFS, record the relative positions of each land cell in the island relative to the starting cell. For example, if the starting cell is at (i, j) and you visit a cell at (i + 2, j - 1), record the coordinate (2, -1). -
Mark Visited Cells:
As you traverse the island, mark the cells as visited (or change them to 0) to avoid reprocessing. -
Store the Island Shape:
Convert the list of relative positions to an immutable object (like a tuple of tuples) and insert it into a set. This ensures that duplicate shapes are not counted more than once. -
Count Distinct Shapes:
The number of elements in the set is the number of distinct island shapes.
Walkthrough Example
Consider the grid:
[
[1,1,0,0],
[1,0,0,0],
[0,0,1,1],
[0,0,1,1]
]
- Start at cell (0,0):
- Begin DFS with starting point (0,0). Record (0,0) for cell (0,0).
- Visit (0,1): record (0,1).
- Visit (1,0): record (1,0).
- The resulting relative shape is {(0,0), (0,1), (1,0)}.
- Later, when processing the bottom-right island, start at (2,2):
-
Record (0,0) for cell (2,2), (0,1) for (2,3), (1,0) for (3,2), and (1,1) for (3,3).
-
The shape is {(0,0), (0,1), (1,0), (1,1)}.
-
- If islands can be translated to match, you will only count those with identical relative positions. In this example, if both islands had the same relative structure, they would count as one. (The given example shows distinct shapes, so the set size would reflect the correct count.)
Code Implementation
Python Code
Java Code
Complexity Analysis
-
Time Complexity:
In the worst case, each cell is visited exactly once by the DFS. If there are ( m \times n ) cells, the time complexity is O(m * n). -
Space Complexity:
-
The recursion stack in DFS can go as deep as O(m * n) in the worst-case scenario (if the entire grid is one big island).
-
The set storing the island shapes may also use up to O(m * n) space in the worst case if every island is distinct.
-
Step-by-Step Walkthrough and Visual Example
For a simple grid:
[
[1,1,0],
[1,0,0],
[0,0,1]
]
- Start at (0,0):
-
Begin DFS with base point (0,0). Record relative position (0,0).
-
Move right to (0,1): record (0,1).
-
Move down from (0,0) to (1,0): record (1,0).
-
The relative positions for this island are {(0,0), (0,1), (1,0)}.
-
- Next, at (2,2):
- Begin DFS with base point (2,2) and record (0,0).
- No adjacent land cells are found.
- The relative shape is {(0,0)}.
- After processing, the set of island shapes contains two entries. Thus, the output is 2.
Common Mistakes and Edge Cases
Common Mistakes
-
Not Recording Relative Positions Correctly:
Failing to normalize the island shape by subtracting the base coordinates leads to different representations for the same shape. -
Visiting a Cell Multiple Times:
Always mark cells as visited (for example, by setting them to 0) to avoid processing the same cell again. -
Handling Grid Boundaries:
Make sure your DFS does not go out of the grid bounds.
Edge Cases
-
Empty Grid:
If the grid is empty or has no land cells (all 0’s), the output should be 0. -
Single Cell Island:
An island consisting of one cell should be handled correctly and represented as a unique shape.
Variations
-
Distinct Islands II:
A variation where two islands are considered the same if one can be rotated or reflected to match the other. -
Max Area of Island:
Instead of counting distinct shapes, determine the largest area of an island.
Related Problems for Further Practice
-
Number of Islands:
Count the total number of islands without considering their shapes. -
Island Perimeter:
Compute the perimeter of an island in a grid. -
Connected Components in a Graph:
A general graph traversal problem similar to counting islands.
GET YOUR FREE
Coding Questions Catalog
