给你一个 m x n 的二进制矩阵 mat。
每一步,你可以选择一个单元格并将它反转(反转表示 0 变 1 ,1 变 0 )。如果存在和它相邻的单元格,那么这些相邻的单元格也会被反转。(注:相邻的两个单元格共享同一条边。)
请你返回将矩阵 mat 转化为全零矩阵的最少反转次数,如果无法转化为全零矩阵,请返回 -1 。
二进制矩阵的每一个格子要么是 0 要么是 1 。
全零矩阵是所有格子都为 0 的矩阵。
示例 1:
输入:mat = [[0,0],[0,1]] 输出:3 解释:一个可能的解是反转 (1, 0),然后 (0, 1) ,最后是 (1, 1) 。
示例 2:
输入:mat = [[0]] 输出:0 解释:给出的矩阵是全零矩阵,所以你不需要改变它。
示例 3:
输入:mat = [[1,1,1],[1,0,1],[0,0,0]] 输出:6
示例 4:
输入:mat = [[1,0,0],[1,0,0]] 输出:-1 解释:该矩阵无法转变成全零矩阵
提示:
来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/minimum-number-of-flips-to-convert-binary-matrix-to-zero-matrix
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
请参考:滑动谜题
class Solution(object):
def minFlips(self, mat):
"""
:type mat: List[List[int]]
:rtype: int
"""
m, n, mat = len(mat), len(mat[0]), "".join(["".join(map(str, row)) for row in mat])
result = "0" * (m * n)
duplicate = set()
queue = [(mat, 0)]
while queue:
mat, count = queue.pop(0)
if mat == result:
return count
for child in self.get_children(mat, m, n):
if child in duplicate:
continue
duplicate.add(child)
queue.append((child, count + 1))
else:
return -1
def get_children(self, mat, m, n):
for ind in range(len(mat)):
row = ind / n
column = ind % n
# convert ind first
s = mat[:ind] + self.convert(mat[ind]) + mat[ind+1:]
left = self.get_left(column, ind)
if left != -1:
s = s[:left] + self.convert(mat[left]) + s[left+1:]
right = self.get_right(column, ind, n)
if right != -1:
s = s[:right] + self.convert(mat[right]) + s[right+1:]
up = self.get_up(row, ind, n)
if up != -1:
s = s[:up] + self.convert(mat[up]) + s[up+1:]
down = self.get_down(row, ind, m, n)
if down != -1:
s = s[:down] + self.convert(mat[down]) + s[down+1:]
yield s
def convert(self, v):
if v == "1":
return "0"
return "1"
def get_left(self, column, ind):
if column == 0:
return -1
return ind - 1
def get_right(self, column, ind, n):
if column == n - 1:
return -1
return ind + 1
def get_up(self, row, ind, n):
if row == 0:
return -1
return ind - n
def get_down(self, row, ind, m, n):
if row == m - 1:
return -1
return ind + n