787. The Maze

https://www.cnblogs.com/grandyang/p/6381458.html

与number of island不一样,递归的函数返回值是bool,不是void。

maze = -1用来表示已经访问的节点。

dp用来记录每个位置的是否能访问,如果dp != -1,就表示这个地方已经访问过了,可以避免多余的访问。

一直滑动用while循环来做,这里并没有没移动一次就增加一个访问。

int x = i,y = j必须这样写,因为之后的4种迭代都是从i、j这个位置出发,x、y在每一次迭代过程中已经发生了变化。

vector的初始化用{},如果是vector<vector<int>>,初始化用{{},{},{}}

class Solution {
public:
/**
* @param maze: the maze
* @param start: the start
* @param destination: the destination
* @return: whether the ball could stop at the destination
*/
bool hasPath(vector<vector<int>> &maze, vector<int> &start, vector<int> &destination) {
// write your code here
int m = maze.size();
if(m <= )
return false;
int n = maze[].size();
if(n <= )
return false;
vector<vector<int>> dp(m,vector<int>(n,-));
return hasPath(maze,dp,start[],start[],destination[],destination[]);
}
bool hasPath(vector<vector<int>>& maze,vector<vector<int>>& dp,int i,int j,int di,int dj){
if(i == di && j == dj)
return true;
if(dp[i][j] != -)
return dp[i][j];
maze[i][j] = -;
int m = maze.size(),n = maze[].size();
bool res = false;
for(auto dir : dirs){
int x = i,y = j;
while(x >= && x < m && y >= && y < n && maze[x][y] != ){
x += dir[];
y += dir[];
}
x -= dir[];
y -= dir[];
if(maze[x][y] != -)
res |= hasPath(maze,dp,x,y,di,dj);
}
return res;
}
vector<vector<int>> dirs{{,-},{,},{,},{-,}};
};

自己写了一遍:

因为一直移动,所以需要一直进行dir的移动计算,直到不满足条件。

注意,这里

maze[new_x][new_y] != 1

而不是写的==0,对于-1的位置,也就是已经遍历过的点,你还是可以继续经过这里去其他地方。这个-1更多的是表示从这个点出发已经访问过了。

class Solution {
public:
/**
* @param maze: the maze
* @param start: the start
* @param destination: the destination
* @return: whether the ball could stop at the destination
*/
bool hasPath(vector<vector<int>> &maze, vector<int> &start, vector<int> &destination) {
// write your code here
if(maze.empty())
return false;
if(maze[].empty())
return false;
if(start.empty() || destination.empty())
return false;
return hasPath(maze,start[],start[],destination[],destination[]);
}
bool hasPath(vector<vector<int>>& maze,int x,int y,int destination_x,int destination_y){
if(x == destination_x && y == destination_y)
return true;
maze[x][y] = -;
bool flag = false;
for(int i = ;i < dirs.size();i++){
int new_x = x;
int new_y = y;
while(new_x >= && new_x < maze.size() && new_y >= && new_y < maze[].size() && maze[new_x][new_y] != ){
new_x += dirs[i][];
new_y += dirs[i][];
}
new_x -= dirs[i][];
new_y -= dirs[i][];
if(maze[new_x][new_y] != -)
flag |= hasPath(maze,new_x,new_y,destination_x,destination_y);
}
return flag;
}
vector<vector<int>> dirs{{-,},{,},{,-},{,}};
};

788. The Maze II

思路:用一个矩阵记录到每个位置的最短距离,初始化为INT_MAX,如果终点到最后仍然为INT_MAX,则表明不可达

错误解法一:

这个解法使用了visited数组,表示已经被访问过,这容易造成有些地方不可达。如果从另一个方向到当前位置

class Solution {
public:
/**
* @param maze: the maze
* @param start: the start
* @param destination: the destination
* @return: the shortest distance for the ball to stop at the destination
*/
int shortestDistance(vector<vector<int>> &maze, vector<int> &start, vector<int> &destination) {
// write your code here
int m = maze.size();
if(m <= )
return -;
int n = maze[].size();
if(n <= )
return -;
if(maze[start[]][start[]] == || maze[destination[]][destination[]] == )
return -;
vector<vector<int>> distance(m,vector<int>(n,INT_MAX));
vector<vector<bool>> visited(m,vector<bool>(n,false));
queue<pair<int,int>> q;
q.push(make_pair(start[],start[]));
distance[start[]][start[]] = ;
while(!q.empty()){
int x = q.front().first;
int y = q.front().second;
q.pop();
visited[x][y] = true;
for(auto dir : dirs){
int x_new = x + dir[];
int y_new = y + dir[];
int des = distance[x][y];
while(x_new >= && x_new < m && y_new >= && y_new < n && !visited[x_new][y_new] && maze[x_new][y_new] == ){
x_new += dir[];
y_new += dir[];
des++;
}
x_new -= dir[];
y_new -= dir[];
if(des < distance[x_new][y_new]){
distance[x_new][y_new] = des;
if(x_new != destination[] || y_new != destination[])
q.push(make_pair(x_new,y_new));
}
}
}
return distance[destination[]][destination[]] == INT_MAX ? - : distance[destination[]][destination[]];
}
vector<vector<int>> dirs{{-,},{,-},{,},{,}};
};

错误解法二:

class Solution {
public:
/**
* @param maze: the maze
* @param start: the start
* @param destination: the destination
* @return: the shortest distance for the ball to stop at the destination
*/
int shortestDistance(vector<vector<int>> &maze, vector<int> &start, vector<int> &destination) {
// write your code here
int m = maze.size();
if(m <= )
return -;
int n = maze[].size();
if(n <= )
return -;
if(maze[start[]][start[]] == || maze[destination[]][destination[]] == )
return -;
vector<vector<int>> distance(m,vector<int>(n,INT_MAX));
queue<pair<int,int>> q;
q.push(make_pair(start[],start[]));
distance[start[]][start[]] = ;
while(!q.empty()){
int x = q.front().first;
int y = q.front().second;
q.pop();
int des = distance[x][y];
for(auto dir : dirs){
int x_new = x + dir[];
int y_new = y + dir[];
while(x_new >= && x_new < m && y_new >= && y_new < n && maze[x_new][y_new] == ){
x_new += dir[];
y_new += dir[];
des++;
}
x_new -= dir[];
y_new -= dir[];
if(des < distance[x_new][y_new]){
distance[x_new][y_new] = des;
if(x_new != destination[] || y_new != destination[])
q.push(make_pair(x_new,y_new));
}
}
}
return distance[destination[]][destination[]] == INT_MAX ? - : distance[destination[]][destination[]];
}
vector<vector<int>> dirs{{-,},{,-},{,},{,}};
};

正确解法:

class Solution {
public:
/**
* @param maze: the maze
* @param start: the start
* @param destination: the destination
* @return: the shortest distance for the ball to stop at the destination
*/
int shortestDistance(vector<vector<int>> &maze, vector<int> &start, vector<int> &destination) {
// write your code here
int m = maze.size();
if(m <= )
return -;
int n = maze[].size();
if(n <= )
return -;
if(maze[start[]][start[]] == || maze[destination[]][destination[]] == )
return -;
vector<vector<int>> distance(m,vector<int>(n,INT_MAX));
queue<pair<int,int>> q;
q.push(make_pair(start[],start[]));
distance[start[]][start[]] = ;
while(!q.empty()){
int x = q.front().first;
int y = q.front().second;
q.pop();
for(auto dir : dirs){
int x_new = x + dir[];
int y_new = y + dir[];
int des = distance[x][y];
while(x_new >= && x_new < m && y_new >= && y_new < n && maze[x_new][y_new] == ){
x_new += dir[];
y_new += dir[];
des++;
}
x_new -= dir[];
y_new -= dir[];
if(des < distance[x_new][y_new]){
distance[x_new][y_new] = des;
if(x_new != destination[] || y_new != destination[])
q.push(make_pair(x_new,y_new));
}
}
}
return distance[destination[]][destination[]] == INT_MAX ? - : distance[destination[]][destination[]];
}
vector<vector<int>> dirs{{-,},{,-},{,},{,}};
};

lintcode 787. The Maze 、788. The Maze II 、的更多相关文章

  1. 记录我的 python 学习历程-Day13 匿名函数、内置函数 II、闭包

    一.匿名函数 以后面试或者工作中经常用匿名函数 lambda,也叫一句话函数. 课上练习: # 正常函数: def func(a, b): return a + b print(func(4, 6)) ...

  2. leetcode 198. House Robber 、 213. House Robber II 、337. House Robber III 、256. Paint House(lintcode 515) 、265. Paint House II(lintcode 516) 、276. Paint Fence(lintcode 514)

    House Robber:不能相邻,求能获得的最大值 House Robber II:不能相邻且第一个和最后一个不能同时取,求能获得的最大值 House Robber III:二叉树下的不能相邻,求能 ...

  3. leetcode 263. Ugly Number 、264. Ugly Number II 、313. Super Ugly Number 、204. Count Primes

    263. Ugly Number 注意:1.小于等于0都不属于丑数 2.while循环的判断不是num >= 0, 而是能被2 .3.5整除,即能被整除才去除这些数 class Solution ...

  4. leetcode 344. Reverse String 、541. Reverse String II 、796. Rotate String

    344. Reverse String 最基础的旋转字符串 class Solution { public: void reverseString(vector<char>& s) ...

  5. 代码随想录第八天 |344.反转字符串 、541. 反转字符串II、剑指Offer 05.替换空格 、151.翻转字符串里的单词 、剑指Offer58-II.左旋转字符串

    第一题344.反转字符串 编写一个函数,其作用是将输入的字符串反转过来.输入字符串以字符数组 s 的形式给出. 不要给另外的数组分配额外的空间,你必须原地修改输入数组.使用 O(1) 的额外空间解决这 ...

  6. leetcode 136. Single Number 、 137. Single Number II 、 260. Single Number III(剑指offer40 数组中只出现一次的数字)

    136. Single Number 除了一个数字,其他数字都出现了两遍. 用亦或解决,亦或的特点:1.相同的数结果为0,不同的数结果为1 2.与自己亦或为0,与0亦或为原来的数 class Solu ...

  7. leetcode 112. Path Sum 、 113. Path Sum II 、437. Path Sum III

    112. Path Sum 自己的一个错误写法: class Solution { public: bool hasPathSum(TreeNode* root, int sum) { if(root ...

  8. leetcode 39. Combination Sum 、40. Combination Sum II 、216. Combination Sum III

    39. Combination Sum 依旧与subsets问题相似,每次选择这个数是否参加到求和中 因为是可以重复的,所以每次递归还是在i上,如果不能重复,就可以变成i+1 class Soluti ...

  9. 颜色转换、随机、16进制转换、HSV

    颜色转换.随机.16进制转换.HSV: /** * * *-----------------------------------------* * | *** 颜色转换.随机.16进制转换.HSV * ...

随机推荐

  1. mysql字典取值,列表包含

    SELECT * FROM e where JSON_CONTAINS(json_extract(scope_detail, '$.shop'), '001');

  2. git 如何忽略已经加入到版本控制的文件

    增加 .gitignore 文件,里面添加需要忽略的文件(file_not_wanted): 执行命令 git rm -r --cached .   注意,最后的点.不要省略. 最后重新将所有文件添加 ...

  3. java基础(8)---接口和lambda

    一.接口  接口定义:  接口抽象方法定义: 二.接口实现类的定义.创建.调用 接口需要一个实现类. 接口实现类的定义: 接口实现类的创建和调用:  接口的好处:  不好的写法: 推荐的写法: 接口实 ...

  4. JAVA BIO至NIO演进

    主要阐述点: 1.同步/异步 or  阻塞/非阻塞 2.网络模型演进 3.NIO代码示例 一.同步/异步 or  阻塞/非阻塞 同步/异步:核心点在于是否等待结果返回.同步即调用者必须等到结果才返回, ...

  5. Yum 安装memcached 与缓存清空

    1.安装 root@pts/0 # yum -y install memcached 2.启动服务         root@pts/0 # /etc/init.d/memcached start 3 ...

  6. 有关 Java (jackson包问题 ,MappingJacksonHttpMessageConverter 和 MappingJackson2HttpMessageConverter问题)

    今天这一系列问题吃掉我四个小时,所以现在吸收掉. 一. 整理所有错误信息: 1.错误信息:java.lang.NoClassDefFoundError: Could not initialize cl ...

  7. NOIP 2017 PJ

    T1:水 T2:水 T3:水 T4:水,二分+DP检测+单调队列优化,然而优化写炸了,还没暴力分高 所以爆炸 (民间)100 + 100 + 100 + 10 = 310 GAME OVER

  8. ES 调优查询亿级数据毫秒级返回!怎么做到的?--文件系统缓存

    一道面试题的引入: 如果面试的时候碰到这样一个面试题:ElasticSearch(以下简称ES) 在数据量很大的情况下(数十亿级别)如何提高查询效率? 这个问题说白了,就是看你有没有实际用过 ES,因 ...

  9. Kubernetes 学习19基于canel的网络策略

    一.概述 1.我们说过,k8s的可用插件有很多,除了flannel之外,还有一个流行的叫做calico的组件,不过calico在很多项目中都会有这个名字被应用,所以他们把自己称为project cal ...

  10. YAML_09 脚本调用变量+触发器

    ansible]# vim adhttp2.yml --- - hosts: cache   remote_user: root   vars:     server: httpd   tasks: ...