【算法一则】矩阵置零 【矩阵】【空间复用】

题目

给定一个 m x n 的矩阵,如果一个元素为 0 ,则将其所在行和列的所有元素都设为 0 。请使用 原地 算法。

示例 1:
在这里插入图片描述

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

示例 2:
在这里插入图片描述

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

提示:

m == matrix.length
n == matrix[0].length
1 <= m, n <= 200
-231 <= matrix[i][j] <= 231 - 1

题解

看到这题的时候我们第一时间肯定想申请一个跟原始matrix二维数组一模一样的数组A,遍历matrix来保存而为素组matrix中为0元素的位置。然后再遍历A将martix中对应的行和列的元素置为0。

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 ++) {for (int j = 0; j < matrix[0].length; j ++) {if (row[i] || col[j]) {matrix[i][j] = 0;}}}
}

在这里插入图片描述
看到这种方式空间使用率会很比较高大概O(2mn),这个时候有没有更好的解决办法呢!我用List<Map<String, String>> 包装类型来放为0元素的位置应该会小一点吧? 因为我只需要存特定是0的位置,其他的不为0的位置我不需要存。说干就干,试验一下:

public void setZeroes(int[][] matrix) {List<Map<Integer, Integer>> list = new ArrayList<>();for (int i = 0; i < matrix.length; i ++) {for (int j = 0; j < matrix[0].length; j ++) {if (matrix[i][j] == 0) {HashMap<Integer, Integer> integerHashMap = new HashMap<>();integerHashMap.put(i, j);list.add(integerHashMap);}}}for (Map<Integer, Integer> data : list) {data.forEach((k, v) -> {for (int i = 0; i < matrix.length; i++) {matrix[i][v] = 0;}for (int i = 0; i < matrix[0].length; i++) {matrix[k][i] = 0;}});}
}

在这里插入图片描述
可以看到空间使用率确实是好了一点,但是我们效率变得非常低了,执行时间只击败了17%的用户。而且空间利用率也不是很完美,主要也是包装类型本身也需要占用一些空间。有没有更好的方案呢?我们接着往下看。

进阶:

一个直观的解决方案是使用 O(mn) 的额外空间,但这并不是一个好的解决方案。
一个简单的改进方案是使用 O(m + n) 的额外空间,但这仍然不是最好的解决方案。
你能想出一个仅使用常量空间的解决方案吗?

public void setZeroes(int[][] matrix) {boolean firstRowHasZero = false;boolean firstColHasZero = false;// 检查第一行是否有零for (int i = 0; i < matrix[0].length; i++) {if (matrix[0][i] == 0) {firstRowHasZero = true;break;}}// 检查第一列是否有零for (int i = 0; i < matrix.length; i++) {if (matrix[i][0] == 0) {firstColHasZero = true;break;}}// 使用第一行和第一列记录零的位置for (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++) {for (int j = 1; j < matrix[0].length; j++) {if (matrix[i][0] == 0 || matrix[0][j] == 0) {matrix[i][j] = 0;}}}// 如果第一行有零,则将第一行置零if (firstRowHasZero) {for (int i = 0; i < matrix[0].length; i++) {matrix[0][i] = 0;}}// 如果第一列有零,则将第一列置零if (firstColHasZero) {for (int i = 0; i < matrix.length; i++) {matrix[i][0] = 0;}}
}

在这里插入图片描述

这个优化后的代码使用了两个布尔变量来记录第一行和第一列是否有零。然后,通过遍历矩阵,使用第一行和第一列来记录零的位置。最后,根据这些记录,将对应的行和列置零。最后,根据第一行和第一列的记录,将第一行和第一列本身置零(如果需要)。

通过这种方法,我们只使用了常量级的额外空间来完成操作,将空间复杂度降到了最低。

总结

上述算法是对给定二维矩阵进行优化,以降低空间复杂度的算法。该算法使用常量级的额外空间来实现。

首先,算法通过遍历矩阵,检查第一行和第一列是否存在零,并使用两个布尔变量记录下来。接下来,算法再次遍历矩阵,将零的位置记录在第一行和第一列中,即如果某个元素为零,就将对应的第一行和第一列的元素置零。

然后,算法再次遍历矩阵,根据第一行和第一列的记录,将对应的行和列置零。具体实现时,如果某个元素所在的行的第一列元素或所在的列的第一行元素为零,则将该元素置零。

最后,算法根据最开始记录的两个布尔变量,如果第一行或第一列存在零,则将整行或整列置零。

通过这种优化方法,算法只使用了常量级的额外空间来完成操作,将空间复杂度降到了最低。这种算法的时间复杂度为O(m×n),其中m是矩阵的行数,n是矩阵的列数。

总之,该算法通过巧妙地利用矩阵的第一行和第一列来记录零的位置,并根据这些记录将对应的行和列置零,从而实现了对给定矩阵的优化。这种优化方法在空间复杂度方面非常高效,适用于处理大规模的二维矩阵。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/826526.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

冰达ROS机器人快速使用指南

欢迎来到《冰达ROS机器人极简使用指南》 Q&#xff1a;这份教程适合谁&#xff1f; A&#xff1a;适合完全0基础新手&#xff0c;需要快速跑起来机器人的基本功能。也适合技术大佬需要快速的了解冰达ROS机器人的使用方法。 Q&#xff1a;这份教程内容很少&#xff0c;是不是…

c# 反射的应用

简述&#xff1a; 反射指程序可以访问、检测和修改它本身状态或行为的一种能力。 程序集包含模块&#xff0c;而模块包含类型&#xff0c;类型又包含成员。反射则提供了封装程序集、模块和类型的对象。 您可以使用使用反射获取类型&#xff0c;根据类型来动态创建对象&#…

机器学习基本流程

Jupyter Notebook 代码连接&#xff1a; machine_learning_demo machine_learning_ensembles Step 1: Imports and Configuration import pandas as pd import numpy as np import copy import json import pickle import joblib import lightgbm as lgb import optuna impor…

【24届数字IC秋招总结】正式批面试经验汇总8——豪威、大普微

文章目录 一、豪威-数字电路设计工程师1.1 面试问题二、大普微-数字验证工程师2.1 一面面试问题2.2 二面面试问题2.3 hr面试问题一、豪威-数字电路设计工程师 面试时间:9.6 1.1 面试问题 1、 logic能不能随机产生x和z,如何产生 2、 断言怎么写的,a为高b为高 3、 讲下项目…

数据结构——7.17.2 查找的基本概念、顺序查找和折半查找

7.1&7.2 查找的基本概念、顺序查找和折半查找 1. 基本概念 1. 关键字&#xff1a;数据元素中标识该元素的某个数据项的值&#xff0c;使用基于关键字的查找&#xff0c;查找结果应该是唯一的。2. 查找表的常见操作1. 查找符合条件的数据元素&#xff1a;查得快即可2. …

国产数据库实践:亚信安慧AntDB在DTC 2024展示创新实力

4月12至13日&#xff0c;我国数据库行业最具影响力的活动之一——第十三届『数据技术嘉年华』(DTC 2024) 在京成功举办&#xff0c;业内众多专家学者、技术领袖、各行业客户和实力厂商均到场参会。亚信安慧AntDB数据库总架构师洪建辉受邀参与“数据库一体化”专题论坛&#xff…

Stylus精讲:网页设计新境界【写作AI一键生成】

首先&#xff0c;这篇文章是基于笔尖AI写作进行文章创作的&#xff0c;喜欢的宝子&#xff0c;也可以去体验下&#xff0c;解放双手&#xff0c;上班直接摸鱼~ 按照惯例&#xff0c;先介绍下这款笔尖AI写作&#xff0c;宝子也可以直接下滑跳过看正文~ 笔尖Ai写作&#xff1a;…

《HCIP-openEuler实验指导手册》1.1Apache安装与测试

一、安装httpd 查看软件仓库中apache版本列表 dnf provides http 安装apache dnf install -y httpd 二、启动http并测试 查看apache版本号 httpd -v 检查配置文件是否正确 httpd -t 启动并设置为开机启动 systemctl enable httpd --now 或 systemctl enable httpd syst…

Android--ConnectivityManager使用

一、前言 Android10之后官方废弃了通过WifiManager连接WIFI的方式&#xff0c;现在要使用ConnectivityManager连接WIFI 二、连接WIFI public class MainActivity extends AppCompatActivity {private static final String TAG"lkx";Overrideprotected void onCrea…

yolov8实战第七天——pyqt5-yolov8实现车牌识别系统(参考论文(约7000字)+环境配置+完整部署代码+代码使用说明+训练好的模型)

基于 pyqt5-yolov8实现车牌识别系统,包括图片车牌识别,视频车牌识别,视频流车牌识别。 效果展示(图片检测,检测到的内容添加到历史记录): 效果展示(视频检测,视频车辆只会添加一条记录,下文更多实际应用中的优化策略): 基于YOLOv8和PyQt5的车牌识别系统设计与…

day60 接雨水 柱状图中的最大矩形

题目1&#xff1a;42 接雨水 题目链接&#xff1a;42 接雨水 题意 n个宽度为1高度不一致的柱子排列&#xff0c;可以接多少雨水 找寻当前柱子的左面第一个比该柱子高的(栈顶的下一个元素)&#xff0c;右面第一个比该柱子高的(当前遍历的元素)&#xff0c;作差 得到宽度&…

CUDA_cudaFree_释放Stream_cudaError_t 错误类型码解释

官方网站 &#xff1a; CUDA Runtime API :: CUDA Toolkit Documentation cudaFree() 说明 cudaFree() 是 CUDA 中用于释放由 cudaMalloc() 或 cudaMallocManaged() 分配的设备内存的函数。它的参数是一个指向设备内存的指针&#xff0c;用于指示要释放的内存块的起始地址。…

在React项目中试用Tailwind

TailwindCSS TailwindCSS 是一个套 CSS 的工具类&#xff0c;把常用的功能都进行了定义&#xff0c;下面是一个官网的例子&#xff0c;可以看到Tailwind对一元页面素写了很多类&#xff0c;日常开发中只要定义一两个类就可以搞定类似的功能了。这里写了这么多 p-6 max-w-sm mx…

java导出数据到excel表中

java导出数据到excel表中 环境说明项目结构1.controller层2.service层3.实现层4.工具类&#xff1a;ExcelUtil.java5.ProductModel.java类 使用的Maven依赖postman请求展示&#xff0c;返回内容需要前端接收浏览器接收说明&#xff08;如果下载下来的为zip类型&#xff0c;记得…

win docker clickhouse 挂载本地目录到容器后无法写入数据问题解决

win docker 部署clickhouse 挂载本地目录到容器后无法写入数据问题具体错误提示代码尝试在docker compose 文件中添加文件操作的许可进一步在docker compose 中配置 ulimits参数如下修改映射的本地目录到d盘最后使用docker的数据卷来映射到容器内部目录&#xff0c;解决了不能…

MySQL查询优化(学习)

1.在MySQL中&#xff0c;如何定位慢查询? MySQL自带慢日志 2.一个SQL语句执行很慢, 如何分析&#xff1f; 可以采用EXPLAIN 或者 DESC命令获取 MySQL 如何执行 SELECT 语句的信息 3.字段分析 idselect_typetablepartitionstype这条sql的连接的类型&#xff0c;性能有好到…

【报错】ModuleNotFoundError: No module named ‘mmcv‘

【报错】&#x1f494;&#x1f494;&#x1f494; ModuleNotFoundError: No module named mmcv 【解决方法】&#x1f49c;&#x1f49c;&#x1f49c; pip3 install mmcv -i https://pypi.tuna.tsinghua.edu.cn/simple 【测试】&#x1f49a;&#x1f49a;&#x1f49a; p…

C语言读取数据检索存档《C语言程序设计》·第6章·用数组处理批量数据

C数组使用 添加链接描述 C语言读取数据检索存档 1 添加链接描述 2 添加链接描述 3 添加链接描述 4 添加链接描述 5 添加链接描述 6 添加链接描述 7 matlab转C 添加链接描述

【从浅学到熟知Linux】基础IO第三弹=>文件系统介绍、软链接与硬链接(含磁盘结构、文件系统存储原理、软硬链接的创建、原理及应用详解)

&#x1f3e0;关于专栏&#xff1a;Linux的浅学到熟知专栏用于记录Linux系统编程、网络编程等内容。 &#x1f3af;每天努力一点点&#xff0c;技术变化看得见 文章目录 理解文件系统物理角度认识磁盘逻辑角度认识磁盘磁盘寻址磁盘中的寄存器 磁盘存储管理 软链接与硬链接软链接…

NVIDIA安装程序失败-Nsight Visual Studio Edition失败解决办法

博主是要升级cuda版本&#xff0c;那么在安装新版本之前需要卸载以前的版本。 博主一溜卸载下去&#xff0c;最后有这么个东西卸载不掉&#xff0c;Nsight Visual Studio Edition 不管是电脑系统卸载还是360卸载&#xff0c;都卸载不掉。 此时安装新的cuda也遇到了这个问题 由…