题目链接

864. 获取所有钥匙的最短路径

题意

给定起点,要求在最短步骤内收集完所有钥匙,遇到每把锁之前只有 有对应的钥匙才能够打开

思路

BFS+状态压缩典型题目

先确定起点和总的钥匙数目,其次难点有两处:

  • 如何确定当前路径下已经收集好特定的钥匙
  • 如何确定钥匙已经全部收集完成

第一个问题:可以把每一个节点的状态定义为(x,y,state),其中state为钥匙数目的二进制表示,例如现在收集'b'这把钥匙,那么state更新为2(000010),括号里面为钥匙的二进制数,如果下一轮遇到'B',只需要检查state里面的倒数第二位是否已经更新即可

第二个问题:钥匙全部收集好的话,那么statey应该为(1<<cnt)-1,其中cnt为总的钥匙数量,比如有6把钥匙,全部收集满的状态为2^6-1(111111),括号里面为钥匙的二进制数

class Solution {
public:
//怎么记录更新之后的连续状态呢
//需要学习的是并非整个路径的状态 ,而是需要记录到某个节点的状态[x][y][state] //1. 有时候可能需要走回头路? 那标记的时候除了位置,再把状态也给标记上
//2. 怎么判断钥匙已经全部取完? (1<<max_len-1) 全部标记
struct Node{
int x,y,state;
Node(int x0,int y0,int state0){
x=x0;y=y0;state=state0;
}
};
int dx[4]={1,-1,0,0};
int dy[4]={0,0,1,-1};
// unordered_set<Node> st;
int st[31][31][129];
queue<Node> Q; int shortestPathAllKeys(vector<string>& grid) {
memset(st,0,sizeof(st)); if(grid.empty()) return 0;
int n=grid.size();
int m=grid[0].size();//n行m列 int cnt=0,nx=-1,ny=-1;//钥匙数量,初始值的位置
for(int i=0;i<n;i++){
for(int j=0;j<m;j++){
if(grid[i][j]=='@') {nx=i;ny=j;}
else if(grid[i][j]>='a' &&grid[i][j]<='z') cnt++;
}
}
Q.push(Node{nx,ny,0});
st[nx][ny][0]=1; int step=0;//移动次数
while(!Q.empty()){
int len=Q.size();//这一层的节点数
// if(cnt<=0) break;
for(int i=0;i<len;i++){
Node cur=Q.front();
Q.pop();
if(cur.state==(1<<cnt)-1) return step;
for(int k=0;k<4;k++){
int tmpx=cur.x+dx[k];
int tmpy=cur.y+dy[k];
int nState=cur.state; if(tmpx <0 || tmpx >n-1 || tmpy<0 || tmpy>m-1) continue;//0~n-1 这里不是n 而是大于n-1....... if(grid[tmpx][tmpy]=='#') continue;
if(grid[tmpx][tmpy]>='a' && grid[tmpx][tmpy]<='z'){
nState=cur.state|(1<<(grid[tmpx][tmpy]-'a'));
} if(grid[tmpx][tmpy]>='A' && grid[tmpx][tmpy]<='Z'){
// printf("%d %d\n",nState,nState & (1<<(grid[tmpx][tmpy]-'A')));
if((nState & (1<<(grid[tmpx][tmpy]-'A')))==0){
continue;
}
} //表示已经访问过 加上状态
if(!st[tmpx][tmpy][nState]){
st[tmpx][tmpy][nState]=1;
Q.push(Node{tmpx,tmpy,nState});
}
}
}
step++;
}
return -1;
}
};

leetcode 864. 获取所有钥匙的最短路径(BFS,状态压缩)的更多相关文章

  1. HDU1429+bfs+状态压缩

    bfs+状态压缩思路:用2进制表示每个钥匙是否已经被找到.. /* bfs+状态压缩 思路:用2进制表示每个钥匙是否已经被找到. */ #include<algorithm> #inclu ...

  2. BFS+状态压缩 hdu-1885-Key Task

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1885 题目意思: 给一个矩阵,给一个起点多个终点,有些点有墙不能通过,有些点的位置有门,需要拿到相应 ...

  3. BFS+状态压缩 HDU1429

    胜利大逃亡(续) Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total S ...

  4. hdoj 5094 Maze 【BFS + 状态压缩】 【好多坑】

    Maze Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 100000/100000 K (Java/Others) Total Sub ...

  5. ACM/ICPC 之 BFS+状态压缩(POJ1324(ZOJ1361))

    求一条蛇到(1,1)的最短路长,题目不简单,状态较多,需要考虑状态压缩,ZOJ的数据似乎比POj弱一些 POJ1324(ZOJ1361)-Holedox Moving 题意:一条已知初始状态的蛇,求其 ...

  6. poj 1753 Flip Game(bfs状态压缩 或 dfs枚举)

    Description Flip game squares. One side of each piece is white and the other one is black and each p ...

  7. HDU 3247 Resource Archiver (AC自己主动机 + BFS + 状态压缩DP)

    题目链接:Resource Archiver 解析:n个正常的串.m个病毒串,问包括全部正常串(可重叠)且不包括不论什么病毒串的字符串的最小长度为多少. AC自己主动机 + bfs + 状态压缩DP ...

  8. [Swift]LeetCode864. 获取所有钥匙的最短路径 | Shortest Path to Get All Keys

    We are given a 2-dimensional grid. "." is an empty cell, "#" is a wall, "@& ...

  9. HDU1429--胜利大逃亡(续)(BFS+状态压缩)

    Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submission( ...

随机推荐

  1. Vue-组件化,父组件传子组件常见传值方式

    前言 我们都知道vue核心之一:组件化,vue中万物皆组件,组件化我认为应该来至于模块化的设计思想,比如在模块化开发中,一个模块就是一个实现特定功能的独立的文件,有了模块我们就更方便去阅读代码,更方便 ...

  2. jwt 简单基本使用加密解密

    import jwt # 加密 encode_jwt=jwt.encode({'uid':'123'},'密钥123',algorithm='HS256') print(encode_jwt) # 解 ...

  3. Difference between @Bean and @Autowired

    Demo01 1 @SpringBootApplication 2 public class Application { 3 4 @Autowired 5 BookingService booking ...

  4. Servlet中的装饰者模式

    装饰者模式 Decorator模式或者Wrapper模式允许修饰或者封装(在字面意义中,即修改行为)一个对象,即使你没有该对象的源代码或者该对象标识为final. Decorator模式适用于无法继承 ...

  5. 记一次 oracle 数据库在宕机后的恢复

    系统:redhat 6.6 oracle版本: Oracle Database 11g Enterprise Edition Release 11.2.0.1.0 - Production 问题描述: ...

  6. [RoarCTF 2019]Easy Calc

    [RoarCTF 2019]Easy Calc 题目 题目打开是这样的 查看源码 .ajax是指通过http请求加载远程数据. 可以发现有一个calc.php,输入的算式会被传入到这个php文件里,尝 ...

  7. [BUUOJ]刮开有奖reverse

    刮开有奖 这是一个赌博程序,快去赚钱吧!!!!!!!!!!!!!!!!!!!!!!!!!!!(在编辑框中的输入值,即为flag,提交即可) 注意:得到的 flag 请包上 flag{} 提交 1.查壳 ...

  8. os模块和os.path模块常用方法

    今天和大家分享python内置模块中的os模块和os.path模块. 1.什么是模块呢? 在计算机开发过程中,代码越写越多,也就越来越难以维护,所以为了可维护的代码,我们会把函数进行分组,放在不同的文 ...

  9. SecureCRT SSH Linux中不显示彩色 字体颜色、文件夹和文件显示的颜色区别开解决办法

    SecureCRT SSH Linux中不显示彩色 字体颜色.文件夹和文件显示的颜色区别开解决办法 实验环境: 刚开始我的情况是这样的:带颜色的显示不出来,然后还能看到,此处有内容,猜测是Secure ...

  10. Node.js躬行记(5)——定时任务的调试

    最近做一个活动,需要用到定时任务,于是使用了 node-schedule 库. 用法很简单,就是可配置开始.结束时间,以及重复执行的时间点,如下所示,从2020-12-23T09:00:00Z开始,每 ...