传送门

坑很多的一题

这里要感谢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 ;
}

UVA1714 Keyboarding的更多相关文章

  1. UVA1714 Keyboarding(bfs)

    UVA1714 Keyboarding bfs 坑点很多的一题(由于一本通的辣鸡翻译会错题意*n). 1.多组数据 2.如果某方向上没有不同字符光标不会动 我们每次预处理出每个点向四个方向下次到达的点 ...

  2. 【广搜】Keyboarding

    题目描述 给定一个r行c列的在电视上的“虚拟键盘”,通过“上,下,左,右,选择”共5个控制键,你可以移动电视屏幕上的光标来打印文本.一开始,光标在键盘的左上角,每次按方向键,光标总是跳到下一个在该方向 ...

  3. Keyboarding (bfs+预处理+判重优化)

    # #10030. 「一本通 1.4 练习 2」Keyboarding [题目描述] 给定一个 $r$ 行 $c$ 列的在电视上的"虚拟键盘",通过「上,下,左,右,选择」共 $5 ...

  4. Keyboarding(信息学奥赛一本通-T1452)

    [题目描述] 出自 World Final 2015 F. Keyboarding 给定一个 r 行 c 列的在电视上的"虚拟键盘",通过「上,下,左,右,选择」共 5 个控制键, ...

  5. Keyboarding

    题目描述 思路 一开始想先写一个bfs,目标字符串要加上一个'*',表示这是一个换行符,然后一个字母一个字母的找,每次重置一下vis数组,bfs返回的结果再加上1,表示要打印这个字母,结果第一个样例没 ...

  6. loj题目总览

    --DavidJing提供技术支持 现将今年7月份之前必须刷完的题目列举 完成度[23/34] [178/250] 第 1 章 贪心算法 √ [11/11] #10000 「一本通 1.1 例 1」活 ...

  7. Windows-universal-samples学习笔记系列三:Navigation

    Navigation Back Button Master/detail Navigation menu (XAML) Pivot Projection XHR, handling navigatio ...

  8. CSU训练分类

    √√第一部分 基础算法(#10023 除外) 第 1 章 贪心算法 √√#10000 「一本通 1.1 例 1」活动安排 √√#10001 「一本通 1.1 例 2」种树 √√#10002 「一本通 ...

  9. LOJ 一本通一句话题解系列:

    第一部分 基础算法 第 1 章 贪心算法 1):「一本通 1.1 例 1」活动安排:按照结束时间排序,然后扫一遍就可以了. 2):「一本通 1.1 例 2」种树:首先要尽量的往区间重叠的部分种树,先按 ...

随机推荐

  1. sqlserver 查询int类型 in (字符串) 报转换int类型出错的问题

    , , '') ) AS c_departNames FROM t_user AS A LEFT JOIN t_role AS B ON A.c_roleId=B.c_roleId 用 CHARIND ...

  2. android task stack

    http://www.android100.net/html/201402/22/5690.html

  3. nodejs的POST请求

    http://blog.csdn.net/puncha/article/details/9015317 Nodejs 发送HTTP POST请求实例 2013-06-03 17:55 71745人阅读 ...

  4. ROS Learning-027 (提高篇-005 A Mobile Base-03) 控制移动平台 --- Twist 消息

    ROS 提高篇 之 A Mobile Base-03 - 控制移动平台 - Twist 消息 我使用的虚拟机软件:VMware Workstation 11 使用的Ubuntu系统:Ubuntu 14 ...

  5. loj10104 [POI 2008]Blockade

    传送门 分析 我们知道对于一个割点,我们如果去掉它就会使原来的图被分为若干块,则这是我们将所有块包含的点的个数两两相乘即可,而如果不是割点则对于图的连通性没有影响.注意在最后要加上2*(n-1)表示去 ...

  6. 8.View类

    Basic Concepts      在Modle/View 结构中,View从Model中提取数据,并显示给用户.View显示数据的方式不一定与Model中数据排列方式相同,也可能与底层数据结构完 ...

  7. 前端基础 之 CSS

    浏览目录 CSS介绍 CSS语法 CSS的几种引入方式 CSS选择器 CSS属性相关 一.CSS介绍 CSS(Cascading Style Sheet,层叠样式表)定义如何显示HTML元素. 当浏览 ...

  8. kaggle Data Leakage

    What is Data Leakage¶ Data leakage is one of the most important issues for a data scientist to under ...

  9. Java之封装特性

    Java中的三大特性:继承,封装,多态: 其中封装概念:封装是把过程和数据包围起来,对数据的访问只能通过已定义的接口. 面向对象计算始于这个基本概念,即现实世界可以被描绘成一系列完全自治.封装的 对象 ...

  10. Web开发秘方(WEB DEVELOPMENT RECIPES)[47.5MB] PDF扫描版

    不借助插件怎样在移动设备上实现动画效果?怎样快速搭建HTML电子邮箱?怎样制作跨PC和移动设备显示的应用界面?怎样利用最新的JavaScript框架提高应用的响应速度?怎样有效利用CoffeeScri ...