|
52 | 52 |
|
53 | 53 | <!-- 这里可写通用的实现逻辑 -->
|
54 | 54 |
|
| 55 | +**方法一:哈希表** |
| 56 | + |
| 57 | +我们观察发现,如果矩阵中的两行满足以下条件之一,则它们可以通过翻转某些列的方式得到相等的行: |
| 58 | + |
| 59 | +1. 两行的对应位置元素相等,即如果其中一行元素为 $1,0,0,1$,则另一行元素也为 $1,0,0,1$; |
| 60 | +1. 两行的对应位置元素相反,即如果其中一行元素为 $1,0,0,1$,则另一行元素为 $0,1,1,0$。 |
| 61 | + |
| 62 | +我们称满足以上条件之一的两行元素为“等价行”,那么题目所求的答案即为矩阵中最多包含等价行的行数。 |
| 63 | + |
| 64 | +因此,我们可以遍历矩阵的每一行,将每一行转换成第一个元素为 $0$ 的“等价行”。具体做法如下: |
| 65 | + |
| 66 | +- 如果当前行的第一个元素为 $0$,那么当前行的元素保持不变; |
| 67 | +- 如果当前行的第一个元素为 $1$,那么我们将当前行的每个元素进行翻转,即 $0$ 变成 $1$, $1$ 变成 $0$。也就是说,我们将以 $1$ 开头的行翻转成以 $0$ 开头的“等价行”。 |
| 68 | + |
| 69 | +这样一来,我们只需要用一个哈希表来统计转换后的每一行,如果最后哈希表只有一个元素,那么说明我们可以通过翻转行或列,将矩阵中所有的 $1$ 去除。 |
| 70 | + |
| 71 | +时间复杂度 $O(m \times n)$,空间复杂度 $O(m)$。其中 $m$ 和 $n$ 分别是矩阵的行数和列数。 |
| 72 | + |
| 73 | +相似题目: |
| 74 | + |
| 75 | +- [1072. 按列翻转得到最大值等行数](/solution/1000-1099/1072.Flip%20Columns%20For%20Maximum%20Number%20of%20Equal%20Rows/README.md) |
| 76 | + |
55 | 77 | <!-- tabs:start -->
|
56 | 78 |
|
57 | 79 | ### **Python3**
|
58 | 80 |
|
59 | 81 | <!-- 这里可写当前语言的特殊实现逻辑 -->
|
60 | 82 |
|
61 | 83 | ```python
|
62 |
| - |
| 84 | +class Solution: |
| 85 | + def removeOnes(self, grid: List[List[int]]) -> bool: |
| 86 | + s = set() |
| 87 | + for row in grid: |
| 88 | + t = tuple(row) if row[0] == grid[0][0] else tuple(x ^ 1 for x in row) |
| 89 | + s.add(t) |
| 90 | + return len(s) == 1 |
63 | 91 | ```
|
64 | 92 |
|
65 | 93 | ### **Java**
|
66 | 94 |
|
67 | 95 | <!-- 这里可写当前语言的特殊实现逻辑 -->
|
68 | 96 |
|
69 | 97 | ```java
|
| 98 | +class Solution { |
| 99 | + public boolean removeOnes(int[][] grid) { |
| 100 | + Set<String> s = new HashSet<>(); |
| 101 | + int n = grid[0].length; |
| 102 | + for (var row : grid) { |
| 103 | + var cs = new char[n]; |
| 104 | + for (int i = 0; i < n; ++i) { |
| 105 | + cs[i] = (char) (row[0] ^ row[i]); |
| 106 | + } |
| 107 | + s.add(String.valueOf(cs)); |
| 108 | + } |
| 109 | + return s.size() == 1; |
| 110 | + } |
| 111 | +} |
| 112 | +``` |
70 | 113 |
|
| 114 | +### **C++** |
| 115 | + |
| 116 | +```cpp |
| 117 | +class Solution { |
| 118 | +public: |
| 119 | + bool removeOnes(vector<vector<int>>& grid) { |
| 120 | + unordered_set<string> s; |
| 121 | + for (auto& row : grid) { |
| 122 | + string t; |
| 123 | + for (int x : row) { |
| 124 | + t.push_back('0' + (row[0] == 0 ? x : x ^ 1)); |
| 125 | + } |
| 126 | + s.insert(t); |
| 127 | + } |
| 128 | + return s.size() == 1; |
| 129 | + } |
| 130 | +}; |
71 | 131 | ```
|
72 | 132 |
|
73 |
| -### **TypeScript** |
| 133 | +### **Go** |
| 134 | +
|
| 135 | +```go |
| 136 | +func removeOnes(grid [][]int) bool { |
| 137 | + s := map[string]bool{} |
| 138 | + for _, row := range grid { |
| 139 | + t := []byte{} |
| 140 | + for _, x := range row { |
| 141 | + if row[0] == 1 { |
| 142 | + x ^= 1 |
| 143 | + } |
| 144 | + t = append(t, byte(x)+'0') |
| 145 | + } |
| 146 | + s[string(t)] = true |
| 147 | + } |
| 148 | + return len(s) == 1 |
| 149 | +} |
| 150 | +``` |
74 | 151 |
|
75 |
| -<!-- 这里可写当前语言的特殊实现逻辑 --> |
| 152 | +### **TypeScript** |
76 | 153 |
|
77 | 154 | ```ts
|
78 |
| - |
| 155 | +function removeOnes(grid: number[][]): boolean { |
| 156 | + const s = new Set<string>(); |
| 157 | + for (const row of grid) { |
| 158 | + if (row[0] === 1) { |
| 159 | + for (let i = 0; i < row.length; i++) { |
| 160 | + row[i] ^= 1; |
| 161 | + } |
| 162 | + } |
| 163 | + const t = row.join(''); |
| 164 | + s.add(t); |
| 165 | + } |
| 166 | + return s.size === 1; |
| 167 | +} |
79 | 168 | ```
|
80 | 169 |
|
81 | 170 | ### **...**
|
|
0 commit comments