class Solution {
public:
vector<vector<int>> buildMatrix(int k, vector<vector<int>>& rowConditions, vector<vector<int>>& colConditions) {
}
};
2392. 给定条件下构造矩阵
给你一个 正 整数 k
,同时给你:
n
的二维整数数组 rowConditions
,其中 rowConditions[i] = [abovei, belowi]
和m
的二维整数数组 colConditions
,其中 colConditions[i] = [lefti, righti]
。两个数组里的整数都是 1
到 k
之间的数字。
你需要构造一个 k x k
的矩阵,1
到 k
每个数字需要 恰好出现一次 。剩余的数字都是 0
。
矩阵还需要满足以下条件:
0
到 n - 1
之间的下标 i
,数字 abovei
所在的 行 必须在数字 belowi
所在行的上面。0
到 m - 1
之间的下标 i
,数字 lefti
所在的 列 必须在数字 righti
所在列的左边。返回满足上述要求的 任意 矩阵。如果不存在答案,返回一个空的矩阵。
示例 1:
输入:k = 3, rowConditions = [[1,2],[3,2]], colConditions = [[2,1],[3,2]] 输出:[[3,0,0],[0,0,1],[0,2,0]] 解释:上图为一个符合所有条件的矩阵。 行要求如下: - 数字 1 在第 1 行,数字 2 在第 2 行,1 在 2 的上面。 - 数字 3 在第 0 行,数字 2 在第 2 行,3 在 2 的上面。 列要求如下: - 数字 2 在第 1 列,数字 1 在第 2 列,2 在 1 的左边。 - 数字 3 在第 0 列,数字 2 在第 1 列,3 在 2 的左边。 注意,可能有多种正确的答案。
示例 2:
输入:k = 3, rowConditions = [[1,2],[2,3],[3,1],[2,3]], colConditions = [[2,1]] 输出:[] 解释:由前两个条件可以得到 3 在 1 的下面,但第三个条件是 3 在 1 的上面。 没有符合条件的矩阵存在,所以我们返回空矩阵。
提示:
2 <= k <= 400
1 <= rowConditions.length, colConditions.length <= 104
rowConditions[i].length == colConditions[i].length == 2
1 <= abovei, belowi, lefti, righti <= k
abovei != belowi
lefti != righti
原站题解
python3 解法, 执行用时: 160 ms, 内存消耗: 22.7 MB, 提交时间: 2023-06-26 09:59:56
from graphlib import TopologicalSorter class Solution: def buildMatrix(self, k: int, rowConditions: List[List[int]], colConditions: List[List[int]]) -> List[List[int]]: def get_pos(cons: List[List[int]]) -> List[int]: ts = TopologicalSorter() for x in range(1, k + 1): ts.add(x) for x, y in cons: ts.add(y, x) pos = [0] * (k + 1) for i, x in enumerate(ts.static_order()): pos[x] = i return pos try: res = [[0] * k for _ in range(k)] for x, (i, j) in enumerate(zip(get_pos(rowConditions), get_pos(colConditions))): res[i][j] = x return res except: return []
golang 解法, 执行用时: 88 ms, 内存消耗: 7.6 MB, 提交时间: 2023-06-26 09:57:11
func topoSort(k int, edges [][]int) []int { g := make([][]int, k) inDeg := make([]int, k) for _, e := range edges { x, y := e[0]-1, e[1]-1 // 顶点编号从 0 开始,方便计算 g[x] = append(g[x], y) inDeg[y]++ } q := make([]int, 0, k) orders := q // 复用队列作为拓扑序 for i, d := range inDeg { if d == 0 { q = append(q, i) } } for len(q) > 0 { x := q[0] q = q[1:] for _, y := range g[x] { if inDeg[y]--; inDeg[y] == 0 { q = append(q, y) } } } if cap(q) > 0 { return nil } return orders[:k] } func buildMatrix(k int, rowConditions, colConditions [][]int) [][]int { row := topoSort(k, rowConditions) col := topoSort(k, colConditions) if row == nil || col == nil { return nil } pos := make([]int, k) for i, v := range col { pos[v] = i } ans := make([][]int, k) for i, x := range row { ans[i] = make([]int, k) ans[i][pos[x]] = x + 1 } return ans }
java 解法, 执行用时: 10 ms, 内存消耗: 53.3 MB, 提交时间: 2023-06-26 09:56:56
class Solution { int[] topoSort(int k, int[][] edges) { List<Integer>[] g = new ArrayList[k]; Arrays.setAll(g, e -> new ArrayList<>()); var inDeg = new int[k]; for (var e : edges) { int x = e[0] - 1, y = e[1] - 1; // 顶点编号从 0 开始,方便计算 g[x].add(y); ++inDeg[y]; } var order = new ArrayList<Integer>(); var q = new ArrayDeque<Integer>(); for (var i = 0; i < k; ++i) if (inDeg[i] == 0) q.push(i); while (!q.isEmpty()) { var x = q.pop(); order.add(x); for (var y : g[x]) if (--inDeg[y] == 0) q.push(y); } return order.stream().mapToInt(x -> x).toArray(); } public int[][] buildMatrix(int k, int[][] rowConditions, int[][] colConditions) { int[] row = topoSort(k, rowConditions), col = topoSort(k, colConditions); if (row.length < k || col.length < k) return new int[][]{}; var pos = new int[k]; for (var i = 0; i < k; ++i) pos[col[i]] = i; var ans = new int[k][k]; for (var i = 0; i < k; ++i) ans[i][pos[row[i]]] = row[i] + 1; return ans; } }
python3 解法, 执行用时: 88 ms, 内存消耗: 22.8 MB, 提交时间: 2023-06-26 09:56:36
# 拓扑排序 class Solution: def buildMatrix(self, k: int, rowConditions: List[List[int]], colConditions: List[List[int]]) -> List[List[int]]: def topo_sort(edges: List[List[int]]) -> List[int]: g = [[] for _ in range(k)] in_deg = [0] * k for x, y in edges: g[x - 1].append(y - 1) # 顶点编号从 0 开始,方便计算 in_deg[y - 1] += 1 order = [] q = deque(i for i, d in enumerate(in_deg) if d == 0) while q: x = q.popleft() order.append(x) for y in g[x]: in_deg[y] -= 1 if in_deg[y] == 0: q.append(y) return order if len(order) == k else None if (row := topo_sort(rowConditions)) is None or (col := topo_sort(colConditions)) is None: return [] pos = {x: i for i, x in enumerate(col)} ans = [[0] * k for _ in range(k)] for i, x in enumerate(row): ans[i][pos[x]] = x + 1 return ans