【笔记】CF1607F Robot on the Board 2 及相关
题目传送门
记忆化搜索
首先,这题
- 走出地图边界:
这时候只要从边界开始往回标号即可。所以搜索的时候需要把途经的点都按顺序记录下来。
- 走到已经走过并记下答案路径上:
这种情况和走到边界很像,只要从碰到的那个点开始往回标号即可。
- 走到了本次搜索到的路径上,也就是出现了环:
这时,环上的每一个点都可以走遍环上路径,所以标记答案为环的长度。剩下的部分是走向环的,所以从与环接触的点开始,依次往回标号即可。
然后只要对每一个点
因为
细节
首先,
多测不清空, WA 两行泪。特别注意,是搜到边界外面时开始更新答案的,所以边界外面一圈也会有答案,清空的时候要大一圈!
其次,直接写深搜,这种做法会爆栈,MLE on #4 。
考虑到本题情况,不需要回溯,所以其实不用深搜,只要用
Code
#include<bits/stdc++.h>
using namespace std;
int T,n,m,a[2005][2005],ans[2005][2005];
int cnth,ansn,ansx,ansy;
bool vis[2005][2005];
char c;
struct node{
int x,y;
}q[4000005];
int turn(char c){
if(c=='L') return 1;
if(c=='R') return 2;
if(c=='U') return 3;
if(c=='D') return 4;
}
void init(){ //初始化
for(int i=0;i<=n+1;i++){
for(int j=0;j<=m+1;j++) ans[i][j]=vis[i][j]=0;
}
ansn=0;
}
void dfs(int x,int y,int k){
while(1){
if(x<1||y<1||x>n||y>m||ans[x][y]){ //走出边界或已经走到过
for(int i=1;i<k;i++) ans[q[i].x][q[i].y]=ans[x][y]+k-i;
return ;
}
if(vis[x][y]){ //出现环
cnth=0;
for(int i=1;i<k;i++){ //算不在环上的大小
if(x==q[i].x&&y==q[i].y) cnth=i;
}
for(int i=1;i<cnth;i++) ans[q[i].x][q[i].y]=k-i; //更新不在环上的答案
for(int i=cnth;i<k;i++) ans[q[i].x][q[i].y]=k-cnth; //更新环上答案
return ;
}
vis[x][y]=1,q[k].x=x,q[k].y=y,k++; //记录下当前状态并走向下一个状态
if(a[x][y]==1) y--;
else if(a[x][y]==2) y++;
else if(a[x][y]==3) x--;
else if(a[x][y]==4) x++;
}
}
void print(){
for(int i=1;i<=n;i++){
for(int j=1;j<=m;j++){
if(ans[i][j]>ansn){
ansn=ans[i][j];
ansx=i,ansy=j;
}
}
}
printf("%d %d %d\n",ansx,ansy,ansn);
}
void solve(){
scanf("%d%d",&n,&m);
scanf("%c",&c);
for(int i=1;i<=n;i++){
for(int j=1;j<=m+1;j++){
scanf("%c",&c);
a[i][j]=turn(c);
}
}
init();
for(int i=1;i<=n;i++){
for(int j=1;j<=m;j++) dfs(i,j,1);
}
print();
}
int main(){
scanf("%d",&T);
while(T--) solve();
return 0;
}
分类:
笔记
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现
· 25岁的心里话