Skip to content

LeetCode 542 LintCode 974 01 Matrix and LeetCode 1765 Map of Highest Peak

Published: at 06:23 AM

Table of contents

Open Table of contents

Description (01 Matrix)

Link to Question: LeetCode 542, LintCode 974

Given an m x n binary matrix mat, return the distance of the nearest 0 for each cell.

The distance between two cells sharing a common edge is 1.

Example 1

Input: mat = [[0,0,0],[0,1,0],[0,0,0]]
Output: [[0,0,0],[0,1,0],[0,0,0]]

Example 2

Input: mat = [[0,0,0],[0,1,0],[1,1,1]]
Output: [[0,0,0],[0,1,0],[1,2,1]]

Constraints:

Note: This question is the same as 1765: https://leetcode.com/problems/map-of-highest-peak/description/

Description (Map of Highest Peak)

Link to Question: LeetCode 1765

You are given an integer matrix isWater of size m x n that represents a map of land and water cells.

You must assign each cell a height in a way that follows these rules:

Find an assignment of heights such that the maximum height in the matrix is maximized.

Return an integer matrix height of size m x n where height[i][j] is cell (i, j)’s height. If there are multiple solutions, return any of them.

Example 1

Input: isWater = [[0,1],[0,0]]
Output: [[1,0],[2,1]]
Explanation: The image shows the assigned heights of each cell.
The blue cell is the water cell, and the green cells are the land cells.

Example 2

Input: isWater = [[0,0,1],[1,0,0],[0,0,0]]
Output: [[1,1,0],[0,1,1],[1,2,2]]
Explanation: A height of 2 is the maximum possible height of any assignment.
Any height assignment that has a maximum height of 2 while still meeting the rules will also be accepted.

Constraints:

Note: This question is the same as 542: https://leetcode.com/problems/01-matrix/

Hint 1

Set each water cell to be 0. The height of each cell is limited by its closest water cell.

Hint 2

Perform a multi-source BFS with all the water cells as sources.

Idea1

The two questions are essentially the same. We could use one line of code to set the value from 1 to 0 and 0 to 1 for the graph.

We could use dynamic programming to solve the question.

We could process the cells from top-left to bottom-right. This way all the cells on the top and left of the current cell have been processed already. We set the cell value to 1 plus the minimum of the two (top, left).

We could then process from bottom-right to top-left. This time we look at cells on the bottom and right of the current cell.

Complexity: Time O(n)O(n), Space O(1)O(1).

Python

class Solution1:
    """
    542, 52 ms, 19.65 mb
    1765, 501 ms, 76.47 mb
    """

    def updateMatrix(self, mat: list[list[int]]) -> list[list[int]]:
        m, n = len(mat), len(mat[0])
        INF = m + n
        for r in range(m):
            for c in range(n):
                # mat[r][c] = 1 - mat[r][c]  # uncomment this line for 1765
                if mat[r][c] > 0:
                    top = mat[r - 1][c] if r > 0 else INF
                    left = mat[r][c - 1] if c > 0 else INF
                    mat[r][c] = min(top, left) + 1
        for r in range(m - 1, -1, -1):
            for c in range(n - 1, -1, -1):
                if mat[r][c] > 0:
                    bottom = mat[r + 1][c] if r < m - 1 else INF
                    right = mat[r][c + 1] if c < n - 1 else INF
                    mat[r][c] = min(mat[r][c], bottom + 1, right + 1)
        return mat

Idea2

We could start from the 0 cells and use breadth-first-search (bfs) to explore the graph. Each step, we process all the cells in the queue, increment the step value and set all the cells explored in this step to that value.

Complexity: Time O(n)O(n), Space O(n)O(n).

Python

class Solution2:
    """
    542, 122 ms, 19.20 mb
    1765, 775 ms, 76.98 mb
    """

    def updateMatrix(self, mat: list[list[int]]) -> list[list[int]]:
        m, n = len(mat), len(mat[0])
        DIR = [0, 1, 0, -1, 0]
        q = deque()
        for r in range(m):
            for c in range(n):
                mat[r][c] = 1 - mat[r][c]  # uncomment this line for 1765
                if mat[r][c] == 0:
                    q.append((r, c))
                else:
                    mat[r][c] = -1  # Marked as not processed
        while q:
            r, c = q.popleft()
            for i in range(4):
                nr, nc = r + DIR[i], c + DIR[i + 1]
                if nr < 0 or nr == m or nc < 0 or nc == n or mat[nr][nc] != -1: continue
                mat[nr][nc] = mat[r][c] + 1
                q.append((nr, nc))
        return mat

Previous Post
LeetCode 2017 Grid Game
Next Post
LeetCode 532 LintCode 1187 K-diff Pairs in An Array