【程序员面试金典】01.08.零矩阵
admin
2024-01-17 12:35:12
0

零矩阵

编写一种算法,若M × N矩阵中某个元素为0,则将其所在的行与列清零。

示例 1:

输入:
[
[1,1,1],
[1,0,1],
[1,1,1]
]
输出:
[
[1,0,1],
[0,0,0],
[1,0,1]
]

示例 2:

输入:
[
[0,1,2,0],
[3,4,5,2],
[1,3,1,5]
]
输出:
[
[0,0,0,0],
[0,4,5,0],
[0,3,1,0]
]

题目解法

解法 1

乍一看,这个问题似乎显而易见,直接遍历整个矩阵,只要发现值为0的元素,就将其所在的行和列清零。不过这种方法存在陷阱:在读取被清零的行或列时,读到的都是0,于是所在行和列都会变成0,很快,整个矩阵的所有元素都会变成0。

避开这个陷阱的方法之一是,新建一个矩阵,标记0元素位置,然后,在第二次遍历矩阵时,将0所在的行与列清零。这种做法的空间复杂度为O(MN)。

实际,不需要用矩阵存储,只需要用两个数组分别记录包含0元素的行和列即可,空间复杂度为O(N)。

算法实现(JAVA)

public class Solution {public void setZeroes(int[][] matrix) {boolean[] row = new boolean[matrix.length];boolean[] col = new boolean[matrix[0].length];for (int i = 0; i < matrix.length; i++) {for (int j = 0; j < matrix[0].length; j++) {if (matrix[i][j] == 0) {row[i] = true;col[j] = true;}}}// 置空行for (int i = 0; i < matrix.length; i++) {if (row[i]) {nullifyRow( matrix, i);}}// 置空列for (int j = 0; j < matrix[0].length; j++) {if (col[j]) {nullifyCol(matrix, j);}}}private void nullifyRow(int[][] matrix, int row) {for (int j = 0; j < matrix[0].length; j++) {matrix[row][j] = 0;}}private void nullifyCol(int[][] matrix, int col) {for (int i = 0; i < matrix.length; i++) {matrix[i][col] = 0;}}
}

解法 2

为了提高空间利用率,可以选用位向量替代布尔数组。通过使用矩阵的第一行代替解法1中的row数组,矩阵的第一列代替col数组,可以将算法的空间复杂度降为O(1),具体步骤如下。

  1. 检查第一行和第一列是否存在0元素,并根据结果设置rowHasZero和colHasZero的值。
  2. 遍历矩阵其余元素,如果matrix[i][j]为0,则将matrix[i][0]和matrix[0][j]置为0。
  3. 遍历矩阵中的其余元素,如果matrix[i][0]为0,则将第i行清零。
  4. 遍历矩阵中的其余元素,如果matrix[0][j]为0,则将第j行清零。
  5. 根据第1步的结果,如果需要则将第一行和第一列清零。

算法实现(JAVA)

public class Solution2 {public void setZeroes(int[][] matrix) {boolean rowHasZero = false;boolean colHasZero = false;// 检查第一行是否有0for (int j = 0; j < matrix[0].length; j++) {if (matrix[0][j] == 0) {rowHasZero = true;break;}}// 检查第一列是否有0for (int i = 0; i < matrix.length; i++) {if (matrix[i][0] == 0) {colHasZero = true;break;}}// 检查数组其余元素是否有0for (int i = 1; i < matrix.length; i++) {for (int j = 1; j < matrix[0].length; j++) {if (matrix[i][j] == 0) {matrix[i][0] = 0;matrix[0][j] = 0;}}}// 根据第一列的值置空行for (int i = 1; i < matrix.length; i++) {if (matrix[i][0] == 0) {nullifyRow(i, matrix);}}// 根据第一行的值置空列for (int j = 1; j < matrix[0].length; j++) {if (matrix[0][j] == 0) {nullifyCol(j, matrix);}}// 置空第一行if (rowHasZero) {nullifyRow(0, matrix);}// 置空第一列if (colHasZero) {nullifyCol(0, matrix);}}private void nullifyRow(int row, int[][] matrix) {for (int j = 0; j < matrix[0].length; j++) {matrix[row][j] = 0;}}private void nullifyCol(int col, int[][] matrix) {for (int i = 0; i < matrix.length; i++) {matrix[i][col] = 0;}}
}

[1] 盖尔.拉克曼.麦克道尔.程序员面试金典(第6版)[M].北京:人民邮电出版社,2019.9:167-169
[2] leetcode.面试题 01.08. 零矩阵

相关内容

热门资讯

原创 4... 写在文章前的声明:在本文之前的说明:本文中所列的投资信息,只是一个对基金资产净值进行排行的客观描述,...
胜宏科技港股大涨49% 做完英... 记者 陈月芹 4月21日,全球AI算力板龙头胜宏科技(02476.HK)登陆港交所,上市首日股价大涨...
永赢基金:聚焦“科技新锐”,科... 数据来源:Wind,时间统计区间为2025/1/1-2026/4/21,指数过往表现不预示未来,不构...
五大阅读趋势显现!当当网发布2... 在第31个世界读书日即将来临之际及首个全民阅读活动周期间,当当网正式发布2026国民阅读洞察报告。 ...
业绩逐季回暖 老百姓大药房一季... 上证报中国证券网讯(记者 夏子航)4月22日晚,老百姓大药房发布2025年年报和2026年一季报。今...
中国20强城市大洗牌:苏州接近... 中国的城市经济竞争格局一直在变化,每年发布的GDP数据都会对城市经济实力进行重新排列。2025年榜又...
直击金宏气体股东会:预期年内氦... 《科创板日报》4月22日讯(记者 郭辉)金宏气体日前举行2025年度股东大会。会上该公司审议了公司年...
5月1日起,俄据悉将叫停哈萨克... 据行业消息人士透露,俄罗斯将于5月1日起停止经友谊管道转运哈萨克斯坦输往德国的石油,相关调整计划已送...
深化具身智能生态布局 京东携手... 4 月 22 日,京东与国内消费级人形机器人头部企业松延动力正式达成三年期战略合作。双方将围绕产品研...
原创 帮... 先问你一个问题,美伊停火今晚到期,按常理避险情绪该升温,黄金应该涨吧?结果恰恰相反——原油涨了,黄金...
300295、600889,将... 三六五网、南京化纤,将被*ST。 公司股票自4月23日开市起停牌一天,于4月24日开市起复牌并实施退...
能源大变天!外媒:羡慕中国的石... 这一次油价突破 110 美元的能源危机,着实魔幻。如果放在十年前,没人会相信中国能在这场风波中获利,...
黄金涨跌两难,现在还能上车吗? 中新网4月22日电(记者 左雨晴) 四月以来,美伊局势反复拉扯,美联储降息预期一变再变。黄金价格在4...
“我身体健康”,库克现身员工大... 当地时间4月21日,受苹果官宣CEO换届影响,公司股价盘中下探超2%,总市值失守4万亿美元关口,收盘...
库克留下一个悬念 工程师能否拯救创新节奏? 听筒Tech(ID:tingtongtech)原创 文 | 赵 森 ...
探索消费信贷与社交支付深度融合... 腾讯这一金融产品再添新功能,4月19日,北京商报记者注意到,微信分付灰度测试转账功能引发热议,在向微...
土耳其主要银行股指早盘下跌2% 每经AI快讯,4月20日,土耳其主要银行股指早盘下跌2%。 每日经济新闻
好用的OTA代运营源头厂家 在如今竞争激烈的酒旅行业中,OTA代运营服务成为了众多酒店、民宿提升竞争力的关键。但市场上的代运营厂...
成都五一出游全国热门第三 “五一”假期临近,同程旅行最新发布的《2026“五一”旅行趋势报告》显示,今年“五一”期间成都同时位...