传送门

坑很多的一题

这里要感谢crk大佬提前帮我把所有的坑都踩了一遍...233

讲一下题目的意思:

给你一个神奇的 r*c 的键盘 (r,c<=50)

上面有大写的字母,数字,' - '号 和 ' * ' 号

有一个光标在键盘上

一开始在左上角,每次可以对光标进行一次操作:

向上,向下,向左,向右 和打印当前光标所在的字符

然后给你一个字符串(保证经过一些操作后可以打出来)

问你最少要几次操作才能打出给定的字符串

坑点:

1.在键盘上可能有很多的位置是同一种字符

2.如果从当前字符向某一个方向走,下一个字符和上一个字符一样

那么就自动跳过,直到找到下一个不一样的字符,而且只算走了一步

3.最后要在键盘上多选择一个 ' * ' 字符,表示回车

4.多组数据

数据这么小,肯定考虑搜索

预处理出每个点向4个方向到达的点

字符转成数字处理

用的是广搜,注意一次只走一步

不能走完了再选(算两步)

要分开来考虑

大概就这样吧,实现起来也不难

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<map>
#include<queue>
using namespace std;
map <char,int> Map;//用map把字符转成数字
int n,m,ans;
int mp[][],xx[]={,,,-},yy[]={,,-,};//mp存键盘状态
int nex[][][][];//nex存每个点向4个方向走,走到的点的横坐标和纵坐标
inline void premap()//预处理map
{
for(int i=;i<=;i++) Map[(char)(''+i-)]=i;
for(int i=;i<=;i++) Map[(char)('A'+i)]=i+;
Map['-']=; Map['*']=;
}
inline void prenex()//预处理nex
{
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
for(int k=;k<;k++)
{
int x=i+xx[k],y=j+yy[k];
while(mp[i][j]==mp[x][y]) x+=xx[k],y+=yy[k];//如果相同就继续走
nex[i][j][k][]=x; nex[i][j][k][]=y;
}
}
struct node
{
int x,y,stp,dis;
//x,y存横纵坐标,stp存当前已经选了几个字符,dis表示进行了几次操作
};//广搜的队列里每个点的内容
int lst[],vis[][],len;
//lst存给定的字符串转成数字后的情况
//vis是记忆化数组,存走到点 i j 时选择的字符最多为多少
inline void bfs()
{
queue <node> q;//每次都重新开一个队列,如果重复用也可以,但是要注意清空
q.push( (node){,,,} );//把初始状态压入队列
while(!q.empty())
{
node u=q.front(); q.pop();
if(mp[u.x][u.y]==lst[u.stp])//如果当前光标所在的字符是当前想要的字符
{
if(u.stp==len)//如果是最后一个字符,就代表找到了最少步数
{
ans=u.dis+;//注意+1
return;
}
//否则
u.stp++; u.dis++;//选择此字符
vis[u.x][u.y]=u.stp;//更新vis
q.push(u);//重新加入队列
}
else//如果不是想要的字符
{
int x,y;
for(int k=;k<;k++)//尝试向4个方向移动
{
x=nex[u.x][u.y][k][]; y=nex[u.x][u.y][k][];
if(x<||x>n||y<||y>m) continue;//判断越界
if(vis[x][y]>=u.stp) continue;//剪枝
vis[x][y]=u.stp;
q.push( (node){x,y,u.stp,u.dis+} );//新状态加入队列
}
}
}
}
char ch[];
int main()
{
premap();
while(scanf("%d",&n)!=EOF)
{
memset(nex,,sizeof(nex));
memset(mp,,sizeof(mp));
memset(lst,,sizeof(lst));
memset(vis,-,sizeof(vis));//注意初始化
ans=;
scanf("%d",&m);
for(int i=;i<=n;i++)
{
scanf("%s",ch);
for(int j=;j<m;j++)
mp[i][j+]=Map[ch[j]];//读入键盘并转成数字
}
scanf("%s",ch); len=strlen(ch);
for(int i=;i<len;i++) lst[i]=Map[ch[i]];//读入字符串并转成数字
lst[len]=;//最后要加一个'*'号
prenex();
bfs();
cout<<ans<<endl;
}
return ;
}

最新文章

  1. 细谈Slick(5)- 学习体会和将来实际应用的一些想法
  2. ASP.NET MVC下的四种验证编程方式
  3. 深入了解 Scala 并发性
  4. (转)Java中的static关键字解析
  5. view的setTag() 和 getTag()应用 (转)
  6. [Node.js] Node + Redis 实现分布式Session方案
  7. 在Linux下搭建SVN服务器
  8. 神、上帝以及老天爷[HDU2048]
  9. Ubuntu 14.04 为 root 帐号开启 SSH 登录
  10. Node.js 学习(四)Node.js 回调函数
  11. Android上的SQLLite性能分析
  12. CSS+DIV布局应用(2015年06月10日)
  13. Ubuntu kylin 有可能成为未来中国的主流系统吗?
  14. Python语言在企业级应用上的十大谬误
  15. 随便讲讲XSS攻击
  16. 1、Sencha cmd学习笔记(一) 使你的sencha cmd跑起来
  17. Android Studio设置快捷键和背景
  18. 最美时光第三方UWP源码公开
  19. 搭建golang学习环境,并用chrome headless获取网页内容
  20. HDU2710-Max Factor-分解质因子

热门文章

  1. android opencv
  2. JAVA基础知识总结6(面向对象特征之一:多态)
  3. 装饰器,装饰器多参数的使用(*arg, **kwargs),装饰器的调用顺序
  4. HotSpot JVM垃圾收集器
  5. 经典的CSS代码(转)
  6. 磨刀——python及相关工具
  7. Executor线程池
  8. 用批处理,批量安装字体文件 (Erector.bat)
  9. MySQL update select组合
  10. 业务逻辑:shiro框架的功能实现