力扣HOT100之矩阵:73. 矩阵置零

这道题我没有想到什么好的办法,直接暴力AC了,直接遍历两次矩阵,第一次遍历用两个向量分别记录出现0的行数和列数,第二次遍历就判断当前的元素的行数或者列数是否出现在之前的两个向量中,若出现了就直接置零,否则就跳过。后面看了下空间复杂度为O(1)的解法,感觉这个思路不错,记录一下。

我们可以将第0行和第0列作为记录的标志位,例如matrix[5][6] == 0,我们就将matrix[0][6]matrix[5][0]置为0,经过一次遍历后,除了第0行和第0列的情况没有统计出来,其余的行和列都已经遍历结束并且打上标签,由于打标签会向第0行和第0列添加0,打完标签后无法判断其本身原本就包含0,因此在遍历矩阵打标签之前,我们需要遍历矩阵的第0行和第0列,如果他们本身就有0,我们就分别用两个变量记录下来,至此,我们先分别遍历了矩阵的第0行和第0列,再遍历了矩阵的其余行和列,这就遍历了一次矩阵,我们再遍历一次矩阵,从下标为1的行和列开始,对于matrix[i][j],如果matrix[0][j] == 0matrix[i][0] == 0,我们就直接将matrix[i][j]置零。遍历结束后,我们还需要对第0行和第0列进行处理,如果他们原本就包含0,则将整行或整列直接置零。这里依然是遍历两次数组,但是由于只进行了比较操作,而没有进行查找操作,程序耗时要比我之前暴力AC短得多。

下面分别是暴力AC代码(空间复杂度O(m + n))和空间复杂度O(1)的代码。

cpp 复制代码
//暴力代码
class Solution {
public:
    void setZeroes(vector<vector<int>>& matrix) {
        vector<int> row;
        vector<int> column;
        for(int i = 0; i < matrix.size(); ++i){
            for(int j = 0; j < matrix[i].size(); ++j){
                if(matrix[i][j] == 0){
                    row.emplace_back(i);
                    column.emplace_back(j);
                }
            }
        }
        for(int i = 0; i < matrix.size(); ++i){
            for(int j = 0; j < matrix[i].size(); ++j){
                if(find(row.begin(), row.end(), i) == row.end() 
                    && find(column.begin(), column.end(), j) == column.end())
                    continue;
                matrix[i][j] = 0;
            }
        }
    }
};
cpp 复制代码
//空间复杂度O(1)代码
class Solution {
public:
    void setZeroes(vector<vector<int>>& matrix) {
        bool row_flag0 = false;   //判断第0行有没有0
        bool column_flag0 = false;  //判断第0列有没有0
        //遍历第0行
        for(int i = 0; i < matrix[0].size(); ++i){
            if(matrix[0][i] == 0){
                row_flag0 = true;
                break;
            }
        }
        //遍历第0列
        for(int i = 0; i < matrix.size(); ++i){
            if(matrix[i][0] == 0){
                column_flag0 = true;
                break;
            }
        }
        //遍历整个矩阵,在第0行和第0列标记0元素的分布情况
        for(int i = 1; i < matrix.size(); ++i){
            for(int j = 1; j < matrix[i].size(); ++j){
                if(matrix[i][j] == 0){
                    matrix[i][0] = 0;  //在第0列的对应位置置零
                    matrix[0][j] = 0;  //在第0行的对应位置置零
                }
            }
        }
        //开始置零
        for(int i = 1; i < matrix.size(); ++i){
            for(int j = 1; j < matrix[i].size(); ++j){
                if(matrix[i][0] == 0 || matrix[0][j] == 0)
                    matrix[i][j] = 0;  
            }
        }
        //判断在打标记之前第0行本身有没有0
        if(row_flag0){
            for(int i = 0; i < matrix[0].size(); ++i)
                matrix[0][i] = 0;
        }
        //判断在打标记之前第0列本身有没有0
        if(column_flag0){
            for(int i = 0; i < matrix.size(); ++i)
                matrix[i][0] = 0;
        }
    }
};
相关推荐
mochensage14 分钟前
C++信息学竞赛中常用函数的一般用法
java·c++·算法
chengooooooo20 分钟前
leetcode Top100 238. 除自身以外数组的乘积|数组系列
算法·leetcode
GUIQU.36 分钟前
【每日一题 | 2025年6.2 ~ 6.8】第16届蓝桥杯部分偏简单题
算法·蓝桥杯·每日一题
weixin_527550401 小时前
初级程序员入门指南
javascript·python·算法
思捻如枫3 小时前
C++数据结构和算法代码模板总结——算法部分
数据结构·c++
嘉陵妹妹3 小时前
深度优先算法学习
学习·算法·深度优先
GalaxyPokemon4 小时前
LeetCode - 53. 最大子数组和
算法·leetcode·职场和发展
小猫咪怎么会有坏心思呢4 小时前
华为OD机考 - 水仙花数 Ⅰ(2025B卷 100分)
数据结构·链表·华为od
hn小菜鸡4 小时前
LeetCode 1356.根据数字二进制下1的数目排序
数据结构·算法·leetcode
zhuiQiuMX4 小时前
分享今天做的力扣SQL题
sql·算法·leetcode