UVA1714 Keyboarding
坑很多的一题
这里要感谢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的更多相关文章
- UVA1714 Keyboarding(bfs)
UVA1714 Keyboarding bfs 坑点很多的一题(由于一本通的辣鸡翻译会错题意*n). 1.多组数据 2.如果某方向上没有不同字符光标不会动 我们每次预处理出每个点向四个方向下次到达的点 ...
- 【广搜】Keyboarding
题目描述 给定一个r行c列的在电视上的“虚拟键盘”,通过“上,下,左,右,选择”共5个控制键,你可以移动电视屏幕上的光标来打印文本.一开始,光标在键盘的左上角,每次按方向键,光标总是跳到下一个在该方向 ...
- Keyboarding (bfs+预处理+判重优化)
# #10030. 「一本通 1.4 练习 2」Keyboarding [题目描述] 给定一个 $r$ 行 $c$ 列的在电视上的"虚拟键盘",通过「上,下,左,右,选择」共 $5 ...
- Keyboarding(信息学奥赛一本通-T1452)
[题目描述] 出自 World Final 2015 F. Keyboarding 给定一个 r 行 c 列的在电视上的"虚拟键盘",通过「上,下,左,右,选择」共 5 个控制键, ...
- Keyboarding
题目描述 思路 一开始想先写一个bfs,目标字符串要加上一个'*',表示这是一个换行符,然后一个字母一个字母的找,每次重置一下vis数组,bfs返回的结果再加上1,表示要打印这个字母,结果第一个样例没 ...
- loj题目总览
--DavidJing提供技术支持 现将今年7月份之前必须刷完的题目列举 完成度[23/34] [178/250] 第 1 章 贪心算法 √ [11/11] #10000 「一本通 1.1 例 1」活 ...
- Windows-universal-samples学习笔记系列三:Navigation
Navigation Back Button Master/detail Navigation menu (XAML) Pivot Projection XHR, handling navigatio ...
- CSU训练分类
√√第一部分 基础算法(#10023 除外) 第 1 章 贪心算法 √√#10000 「一本通 1.1 例 1」活动安排 √√#10001 「一本通 1.1 例 2」种树 √√#10002 「一本通 ...
- LOJ 一本通一句话题解系列:
第一部分 基础算法 第 1 章 贪心算法 1):「一本通 1.1 例 1」活动安排:按照结束时间排序,然后扫一遍就可以了. 2):「一本通 1.1 例 2」种树:首先要尽量的往区间重叠的部分种树,先按 ...
随机推荐
- 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 ...
- android task stack
http://www.android100.net/html/201402/22/5690.html
- nodejs的POST请求
http://blog.csdn.net/puncha/article/details/9015317 Nodejs 发送HTTP POST请求实例 2013-06-03 17:55 71745人阅读 ...
- ROS Learning-027 (提高篇-005 A Mobile Base-03) 控制移动平台 --- Twist 消息
ROS 提高篇 之 A Mobile Base-03 - 控制移动平台 - Twist 消息 我使用的虚拟机软件:VMware Workstation 11 使用的Ubuntu系统:Ubuntu 14 ...
- loj10104 [POI 2008]Blockade
传送门 分析 我们知道对于一个割点,我们如果去掉它就会使原来的图被分为若干块,则这是我们将所有块包含的点的个数两两相乘即可,而如果不是割点则对于图的连通性没有影响.注意在最后要加上2*(n-1)表示去 ...
- 8.View类
Basic Concepts 在Modle/View 结构中,View从Model中提取数据,并显示给用户.View显示数据的方式不一定与Model中数据排列方式相同,也可能与底层数据结构完 ...
- 前端基础 之 CSS
浏览目录 CSS介绍 CSS语法 CSS的几种引入方式 CSS选择器 CSS属性相关 一.CSS介绍 CSS(Cascading Style Sheet,层叠样式表)定义如何显示HTML元素. 当浏览 ...
- kaggle Data Leakage
What is Data Leakage¶ Data leakage is one of the most important issues for a data scientist to under ...
- Java之封装特性
Java中的三大特性:继承,封装,多态: 其中封装概念:封装是把过程和数据包围起来,对数据的访问只能通过已定义的接口. 面向对象计算始于这个基本概念,即现实世界可以被描绘成一系列完全自治.封装的 对象 ...
- Web开发秘方(WEB DEVELOPMENT RECIPES)[47.5MB] PDF扫描版
不借助插件怎样在移动设备上实现动画效果?怎样快速搭建HTML电子邮箱?怎样制作跨PC和移动设备显示的应用界面?怎样利用最新的JavaScript框架提高应用的响应速度?怎样有效利用CoffeeScri ...