[LeetCode] 01 Matrix 题解
题意
思路
我一开始的时候想的是最简单的方法,就是遍历所有的值为1的元素,再根据其为起点进行BFS,计算层数,但是这个方法超时了;
其实,可以不用从1开始遍历,从0开始遍历,找到和值为1相邻的0,将其的层数设置为1就行了,为什么可以不用从1开始,因为并没有要求从规定的起点到指定的位置,计算最小距离,而是计算一整个周围,只要周围存在1,则将其加入到队列,计算相应的距离(又可能存在别多个1包围的1的情况),注意的是,在访问过1的结点后下次不可以再进行计算。
实现
//
//
#include "../PreLoad.h"
class Solution {
public:
/**
* 三重循环,最外层为所有1的结点,里面两层是实现BFS
* 导致时间复杂度过高,待优化
* @param matrix
* @return
*/
vector< vector<int>> layouts = {{-1, 0}, {1, 0}, {0, -1}, {0, 1}};
vector<vector<int>> updateMatrix(vector<vector<int>>& matrix) {
vector<vector<int>> result(matrix);
if (matrix.size() == 0) {
return result;
}
size_t row = matrix.size();
size_t col = matrix[0].size();
deque<pair<int, int>> queues;
vector<vector<int>> visited(row, vector<int>(col , 0));
bool isHaveOne = false;
for (size_t i = 0; i < row; i++) {
for (size_t j = 0; j < col; j++) {
if (matrix[i][j]) {
queues.push_back({i, j});
isHaveOne = true;
}
}
}
if (!isHaveOne) {
return result;
}
while (!queues.empty()) {
auto content = queues.front();
queues.pop_front();
bool found = false;
int level = 0;
deque<pair<int, int>> tqueue;
tqueue.push_back(content);
vector<vector<int>> tvisited(visited);
tvisited[content.first][content.second] = 1;
while (!found && !tqueue.empty()) {
level++;
int queue_len = tqueue.size();
// 保证队列中的每个数都能加上基本的平方数
for (int i = 0; i < queue_len; i++) {
auto tcontent = tqueue.front();
tqueue.pop_front();
for (auto temp : layouts) {
int newx = tcontent.first + temp[0];
int newy = tcontent.second + temp[1];
if (newx < 0 || newy < 0 || newx >= row || newy >= col || tvisited[newx][newy]) {
continue;
}
if (!matrix[newx][newy]) {
found = true;
break;
}
tvisited[newx][newy] = 1;
tqueue.push_back({newx, newy});
}
}
}
tvisited = visited;
result[content.first][content.second] = level;
}
return result;
}
// 做法错误
vector<vector<int>> updateMatrix2(vector<vector<int>>& matrix) {
vector<vector<int>> result(matrix);
if (matrix.size() == 0) {
return result;
}
size_t row = matrix.size();
size_t col = matrix[0].size();
deque<pair<int, int>> queues;
bool isHaveOne = false;
for (size_t i = 0; i < row; i++) {
for (size_t j = 0; j < col; j++) {
if (matrix[i][j]) {
queues.push_back({i, j});
isHaveOne = true;
}
}
}
if (!isHaveOne) {
return result;
}
vector<vector<int>> visited(row, vector<int>(col , 0));
vector<vector<int>> tvisited(visited);
while (!queues.empty()) {
auto content = queues.front();
queues.pop_front();
int level = 0;
bool found = false;
DFSHelper(matrix, result, visited, level, content.first, content.second, row, col, found);
tvisited = visited;
//result[content.first][content.second] = level;
}
return result;
}
// 无法保证取得的路径是最短的,因为是深度递归,所以有可能找的那条路径全都是1的,所以这样使用DFS是错误的
void DFSHelper(vector<vector<int>>& matrix, vector<vector<int>>& result, vector<vector<int>>& visited,
int& dis, int x, int y, int row, int col, bool& found) {
if (found) {
return ;
}
dis++;
visited[x][y] = 1;
if (!matrix[x][y]) {
result[x][y] = dis;
found = true;
return ;
}
else {
for (auto temp : layouts) {
int newx = x + temp[0];
int newy = y + temp[1];
if (found) {
return ;
}
if (newx < 0 || newy < 0 || newx >= row || newy >= col || visited[newx][newy]) {
continue;
}
DFSHelper(matrix, result, visited, dis, newx, newy, row, col, found);
if (found) {
return ;
}
}
}
dis--;
visited[x][y] = 0;
}
/**
* 将二维数组中为0的加入到队列中,1的则置为-1
* 因为必然存在和1的元素相邻的元素0,所以当找到这样的周围是1的0时
* 则把这个1的元素同样加入到队列中,因为可能会存在被1包围的1
* 同时设置其距离,这个则需要将其值设为初始元素的值(同样是1的元素)+1,
* 这个时候其的值不再是-1,同时起到了纪录其已经访问过了的作用
*
* 可以理解为如果上一个(初始位置)如果是0,则说明其在周围,自然为1
* 但是也会碰到被1包围的1,同样根据上面计算出来的1去计算后面的1的元素
* 有些dp的思想
*
* @param matrix
* @return
*/
vector<vector<int>> updateMatrix3(vector<vector<int>>& matrix) {
vector<vector<int>> result(matrix);
if (matrix.size() == 0) {
return result;
}
size_t row = matrix.size();
size_t col = matrix[0].size();
typedef pair<int, int> tp;
deque<pair<int, int>> queues;
for (size_t i = 0; i < row; i++) {
for (size_t j = 0; j < col; j++) {
if (matrix[i][j] == 0) {
queues.push_back(tp(i, j));
}
else {
result[i][j] = -1;
}
}
}
while (!queues.empty()) {
auto content = queues.front();
queues.pop_front();
for (auto temp : layouts) {
int newx = content.first + temp[0];
int newy = content.second + temp[1];
if (newx >= 0 && newx < row && newy >= 0 && newy < col && result[newx][newy] == -1) {
result[newx][newy] = result[content.first][content.second] + 1; //注意不是自增
queues.push_back({newx, newy});
}
}
}
return result;
}
// 计算层数,并将其设置为负数,作为访问过的标记
vector<vector<int>> updateMatrix4(vector<vector<int>>& matrix) {
if (matrix.size() == 0) {
return matrix;
}
size_t row = matrix.size();
size_t col = matrix[0].size();
typedef pair<int, int> tp;
deque<pair<int, int>> queues;
for (size_t i = 0; i < row; i++) {
for (size_t j = 0; j < col; j++) {
if (matrix[i][j] == 0) {
queues.push_back(tp(i, j));
}
}
}
int dis = 0;
while (!queues.empty()) {
dis++;
int queue_len = queues.size();
// 保证队列中的每个数都能加上基本的平方数
for (int i = 0; i < queue_len; i++) {
auto content = queues.front();
queues.pop_front();
for (auto temp : layouts) {
int newx = content.first + temp[0];
int newy = content.second + temp[1];
if (newx >= 0 && newx < row && newy >= 0 && newy < col && matrix[newx][newy] == 1) {
matrix[newx][newy] = -dis; //做标记
queues.push_back({newx, newy});
}
}
}
}
for(int i = 0; i < row; ++i){
for(int j = 0; j < col; ++j)
if(matrix[i][j] < 0) matrix[i][j] = -matrix[i][j];
}
return matrix;
}
void test() {
vector< vector<int>> water = {
{0, 0, 0},
{0, 1, 0},
{0, 0, 0},
};
vector<vector<int>> result = this->updateMatrix4(water);
for (auto i = 0; i < result.size(); i++) {
for (auto j = 0; j < result[0].size(); j++) {
cout << result[i][j] << ", ";
}
cout << endl;
}
}
};
[LeetCode] 01 Matrix 题解的更多相关文章
- [LeetCode] 01 Matrix 零一矩阵
Given a matrix consists of 0 and 1, find the distance of the nearest 0 for each cell. The distance b ...
- [Leetcode] 01 Matrix
问题: https://leetcode.com/problems/01-matrix/#/description 基本思路:广度优先遍历,根据所有最短距离为N的格找到所有距离为N+1的格,直到所有的 ...
- [Leetcode Week10]01 Matrix
01 Matrix 题解 原创文章,拒绝转载 题目来源:https://leetcode.com/problems/01-matrix/description/ Description Given a ...
- 【LeetCode】01 Matrix 解题报告
[LeetCode]01 Matrix 解题报告 标签(空格分隔): LeetCode 题目地址:https://leetcode.com/problems/01-matrix/#/descripti ...
- [leetcode] 542. 01 Matrix (Medium)
给予一个矩阵,矩阵有1有0,计算每一个1到0需要走几步,只能走上下左右. 解法一: 利用dp,从左上角遍历一遍,再从右下角遍历一遍,dp存储当前位置到0的最短距离. 十分粗心的搞错了col和row,改 ...
- leetcode 542. 01 Matrix 、663. Walls and Gates(lintcode) 、773. Sliding Puzzle 、803. Shortest Distance from All Buildings
542. 01 Matrix https://www.cnblogs.com/grandyang/p/6602288.html 将所有的1置为INT_MAX,然后用所有的0去更新原本位置为1的值. 最 ...
- LeetCode: Spiral Matrix II 解题报告-三种方法解决旋转矩阵问题
Spiral Matrix IIGiven an integer n, generate a square matrix filled with elements from 1 to n2 in sp ...
- 计算机学院大学生程序设计竞赛(2015’12)01 Matrix
01 Matrix Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total ...
- hdu 01 Matrix
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submission ...
随机推荐
- IOS解析XML
XML也许是我们储存数据和通讯数据中最常见的一种简易方式,当我们来到XML的海洋时,我们会发现当我们用iPhone程序解析XML时,我们是有如此多的选项,让人眼花缭乱.iOS SDK本身就带有两种不同 ...
- 利用终端命令实现进入ntfs分区有两种方法。
一.手动设置ubuntu自动挂载Windows分区方法:1.先用FDISK命令查看一下磁盘的UUID $sudo fdisk -l /dev/sda1 * 1 851 6835626 83 Linux ...
- [html] 学习笔记-Canvas使用路径
想要绘制其他图形,需要使用路径,使用路径包含4个步骤,开始创建路径.创建图形的路径.路径创建完成后关闭路径.设定绘制样式,之后就可以调用绘制方法绘制路径了. 1.绘制圆形 <!DOCTYPE h ...
- [html5] 学习笔记-html5音频视频
HTML5 最大的新特色之一就是支持音频和视频.在 HTML5 之前,我们必须使用插件如 Silverlight 或 Flash 来实现这些功能.在 HTML5 中,可以直接使用新标签< au ...
- 【小分享】Date对象封装,时间格式化函数time()
今天再来分享下Date的应用知识点 先看效果,类似于php里边的time('yyyy-mm-dd')用法,但是我这里没有完全依照php的参数格式来,如果有需要稍微修改一下就行. 首先,明确需要用到的参 ...
- PLSQL程序流程
IF语句结构: if(条件表达式)- -then- -执行语句;- -end; IF-THEN-ELSE语句结构: if(条件表达式)- -then- -执行语句;- -else- -执行语句;- e ...
- DataReader的用法程序简析
// 2015/07/05 using System; using System.Collections.Generic; using System.Linq; using System.Text; ...
- Spark 键值对RDD操作
键值对的RDD操作与基本RDD操作一样,只是操作的元素由基本类型改为二元组. 概述 键值对RDD是Spark操作中最常用的RDD,它是很多程序的构成要素,因为他们提供了并行操作各个键或跨界点重新进行数 ...
- Unity3d的序列帧动画
马上这星期就要过去了,为了完成每星期写一篇博客的目标,熬夜也要写完. 最近项目中用到了很多序列帧动画,之前看教程也接触过序列帧动画,但当时没用到,就没仔细研究,这次就借着这个机会好好总结一下序列帧动画 ...
- angular、vue使用感受
最近开始学习并使用vue.js,并使用vue+node开发了一个移动端APP来练手,下面想聊聊我对于vue的粗浅看法,并将它和angular进行一些对比: 1.vue是一个轻量.高效的前端组件化框架, ...