This is a neetcode problem that I told Job him and I can do together.
Given a 2D grid grid where '1'
represents land and '0'
represents water, count and return the number of islands.
An island is formed by connecting adjacent lands horizontally or vertically and is surrounded by water. You may assume water is surrounding the grid (i.e., all the edges are water).
Input:
return [
["0","1","1","1","0"],
["0","1","0","1","0"],
["1","1","0","0","0"],
["0","0","0","0","0"]
]
Output: 1
Input:
return [
["1","1","0","0","1"],
["1","1","0","0","1"],
["0","0","1","0","0"],
["0","0","0","1","1"]
]
Output: 4
1 <= grid.length, grid[i].length <= 100
grid[i][j]
is'0'
or'1'
.
As always, we start by looking at hte constraints. Worst case scenario here is 10,000
grid cells. That's not very much
Lets say we just sweep accross all the cells, for each one that's a 1
, we are going to look around and visit all other connected 1
to basically investigate the island. Each time we ru into a previously unvisited 1
that's a new island!
Worst case scenario for that is a checkerboard pattern which would result in visiting something like 10000 + 4*(10000/2) = 30000
cells. That's nothing, optimization beyond that is worthless. Let's just do that.
Since I want to share this with Job, let's do it in python since he knows python.
So first, we're going to need the ability to follow the 1s around and accumulate individual islands
def accumulate(grid, visited_islands, visited_other, coordinate):
if coordinate in visited_islands or coordinate in visited_other:
return
r, c = coordinate
if not (0 <= r < len(grid)) or not (0 <= c < len(grid[r])):
return
if grid[r][c] == "0":
return visited_other.add(coordinate)
visited_islands.add(coordinate)
accumulate(grid, visited_islands, visited_other, (r-1, c))
accumulate(grid, visited_islands, visited_other, (r, c+1))
accumulate(grid, visited_islands, visited_other, (r+1, c))
accumulate(grid, visited_islands, visited_other, (r, c-1))
Lets see if we can use that to get an island
island = set()
accumulate(example1, island, set(), (1, 1))
return island
0 | 1 |
2 | 1 |
1 | 1 |
0 | 3 |
2 | 0 |
0 | 2 |
1 | 3 |
Yes! We can.
Ok, so then lets figure out the islands
def get_islands(grid):
"Yield back one node on each island in the grid of characters. Water is a '0'."
visited = set()
for r, row in enumerate(grid):
for c, cell in enumerate(row):
coordinate = (r,c)
if cell != "0" and (coordinate not in visited):
yield coordinate
accumulate(grid, visited, visited, coordinate)
So that will yield back once for each island. Will it give us the right answers?
return sum((1 for _ in get_islands(example1)))
1
return sum((1 for _ in get_islands(example2)))
4
Why yes! Yes it will.