Codeforces Round #354 (Div. 2)-D
Theseus has just arrived to Crete to fight Minotaur. He found a labyrinth that has a form of a rectangular field of sizen × m and consists of blocks of size 1 × 1.
Each block of the labyrinth has a button that rotates all blocks 90 degrees clockwise. Each block rotates around its center and doesn't change its position in the labyrinth. Also, each block has some number of doors (possibly none). In one minute, Theseus can either push the button in order to rotate all the blocks 90 degrees clockwise or pass to the neighbouring block. Theseus can go from block A to some neighbouring block B only if block A has a door that leads to block B and block B has a door that leads to block A.
Theseus found an entrance to labyrinth and is now located in block (xT, yT) — the block in the row xT and column yT. Theseus know that the Minotaur is hiding in block (xM, yM) and wants to know the minimum number of minutes required to get there.
Theseus is a hero, not a programmer, so he asks you to help him.
The first line of the input contains two integers n and m (1 ≤ n, m ≤ 1000) — the number of rows and the number of columns in labyrinth, respectively.
Each of the following n lines contains m characters, describing the blocks of the labyrinth. The possible characters are:
- «+» means this block has 4 doors (one door to each neighbouring block);
- «-» means this block has 2 doors — to the left and to the right neighbours;
- «|» means this block has 2 doors — to the top and to the bottom neighbours;
- «^» means this block has 1 door — to the top neighbour;
- «>» means this block has 1 door — to the right neighbour;
- «<» means this block has 1 door — to the left neighbour;
- «v» means this block has 1 door — to the bottom neighbour;
- «L» means this block has 3 doors — to all neighbours except left one;
- «R» means this block has 3 doors — to all neighbours except right one;
- «U» means this block has 3 doors — to all neighbours except top one;
- «D» means this block has 3 doors — to all neighbours except bottom one;
- «*» means this block is a wall and has no doors.
Left, right, top and bottom are defined from representing labyrinth as a table, where rows are numbered from 1 to nfrom top to bottom and columns are numbered from 1 to m from left to right.
Next line contains two integers — coordinates of the block (xT, yT) (1 ≤ xT ≤ n, 1 ≤ yT ≤ m), where Theseus is initially located.
Last line contains two integers — coordinates of the block (xM, yM) (1 ≤ xM ≤ n, 1 ≤ yM ≤ m), where Minotaur hides.
It's guaranteed that both the block where Theseus starts and the block where Minotaur is hiding have at least one door. Theseus and Minotaur may be initially located at the same block.
If Theseus is not able to get to Minotaur, then print -1 in the only line of the output. Otherwise, print the minimum number of minutes required to get to the block where Minotaur is hiding.
2 2
+*
*U
1 1
2 2
-1
2 3
<><
><>
1 1
2 1
4
Assume that Theseus starts at the block (xT, yT) at the moment 0.
题意:给定一个图表,然后是开始位置和目标位置,从开始到目标的最少时间,不存在输出-1. 每一秒可以向能走的方向走一步或者顺时针旋转全部按钮90度。注意如果你要从A->B,那么必须B->A才能从A走到B。
思路:BFS,按照思路模拟吧,比较麻烦的一道题目。 开始以为会TLE用的A*发现出来的不一定最优解。改成普通的队列就AC了。
#define _CRT_SECURE_NO_DEPRECATE
#include<iostream>
#include<algorithm>
#include<cstdio>
#include<string>
#include<cstring>
#include<cmath>
#include<bitset>
#include<queue>
#include<vector>
#include<map>
using namespace std;
#define INF 0x3f3f3f3f
#define PI 3.14159
const int MAXN = + ;
char g[MAXN][MAXN];
int dist[][] = { , , , -, , , -, }; //0:right 1:left 2:down 3:up
int n, m, vis[MAXN][MAXN][];
map<char, vector<int> >Move;
struct Node
{
int x, y;
int step;
char str;
int cnt;
int f;
Node(int a = , int b = , int c = , char d = '*', int e = , int ff = ) :x(a), y(b), step(c), str(d), cnt(e), f(ff){};
};
struct cmp
{
bool operator ()(const Node &a, const Node &b)const
{
return a.f>b.f;
}
};
void init() //初始化每个按钮能走的方向
{
vector<int> op1; op1.push_back(); op1.push_back(); op1.push_back(); op1.push_back(); Move['+'] = op1;
vector<int> op2; op2.push_back(); op2.push_back(); Move['-'] = op2;
vector<int> op3; op3.push_back(); op3.push_back(); Move['|'] = op3;
vector<int> op4; op4.push_back(); Move['^'] = op4;
vector<int> op5; op5.push_back(); Move['>'] = op5;
vector<int> op6; op6.push_back(); Move['<'] = op6;
vector<int> op7; op7.push_back(); Move['v'] = op7;
vector<int> op8; op8.push_back(); op8.push_back(); op8.push_back(); Move['L'] = op8;
vector<int> op9; op9.push_back(); op9.push_back(); op9.push_back(); Move['R'] = op9;
vector<int> op10; op10.push_back(); op10.push_back(); op10.push_back(); Move['U'] = op10;
vector<int> op11; op11.push_back(); op11.push_back(); op11.push_back(); Move['D'] = op11;
}
bool check(int x, int y)//判断越界
{
return x >= && x<n&&y >= && y<m;
}
int Get_F(Node a, int ex, int ey) //A*函数
{
return abs(a.x - ex) + abs(a.y - ey) + a.step;
}
char rotates(char str, int cnt) //得到rotates后按钮
{
char ST;
if (str == '-'){ cnt % ? ST = '|' : ST = '-'; }
else if (str == '|'){ cnt % ? ST = '-' : ST = '|'; }
else if (str == '^')
{
if (cnt % == ){ ST = '>'; }
else if (cnt % == ){ ST = 'v'; }
else if (cnt % == ){ ST = '<'; }
else{ ST = '^'; }
}
else if (str == '>')
{
if (cnt % == ){ ST = 'v'; }
else if (cnt % == ){ ST = '<'; }
else if (cnt % == ){ ST = '^'; }
else{ ST = '>'; }
}
else if (str == 'v')
{
if (cnt % == ){ ST = '<'; }
else if (cnt % == ){ ST = '^'; }
else if (cnt % == ){ ST = '>'; }
else{ ST = 'v'; }
}
else if (str == '<')
{
if (cnt % == ){ ST = '^'; }
else if (cnt % == ){ ST = '>'; }
else if (cnt % == ){ ST = 'v'; }
else { ST = '<'; }
}
else if (str == 'L')
{
if (cnt % == ){ ST = 'U'; }
else if (cnt % == ){ ST = 'R'; }
else if (cnt % == ){ ST = 'D'; }
else{ ST = 'L'; }
}
else if (str == 'R')
{
if (cnt % == ){ ST = 'D'; }
else if (cnt % == ){ ST = 'L'; }
else if (cnt % == ){ ST = 'U'; }
else{ ST = 'R'; }
}
else if (str == 'U')
{
if (cnt % == ){ ST = 'R'; }
else if (cnt % == ){ ST = 'D'; }
else if (cnt % == ){ ST = 'L'; }
else{ ST = 'U'; }
}
else if (str == 'D')
{
if (cnt % == ){ ST = 'L'; }
else if (cnt % == ){ ST = 'U'; }
else if (cnt % == ){ ST = 'R'; }
else{ ST = 'D'; }
}
else if (str == '+'){ ST = '+'; }
return ST;
}
bool checkMove(Node b, Node a,int op)//check B can to A?
{
char moveop = rotates(b.str, b.cnt);
if (op == )
{
for (int i = ; i < Move[moveop].size(); i++)
{
if (Move[moveop][i] == )
{
return true;
}
}
}
else if (op == )
{
for (int i = ; i < Move[moveop].size(); i++)
{
if (Move[moveop][i] == )
{
return true;
}
}
}
else if (op == )
{
for (int i = ; i < Move[moveop].size(); i++)
{
if (Move[moveop][i] == )
{
return true;
}
}
}
else
{
for (int i = ; i < Move[moveop].size(); i++)
{
if (Move[moveop][i] == )
{
return true;
}
}
}
return false;
}
int dfs(Node s, Node e)
{
//priority_queue<Node, vector<Node>, cmp >Q;
queue<Node>Q;
memset(vis, , sizeof(vis));
Q.push(s);
vis[s.x][s.y][s.cnt % ] = ;
while (!Q.empty())
{
Node Top = Q.front(); Q.pop();
//Node Top = Q.top(); Q.pop();
if (Top.x == e.x&&Top.y == e.y)
{
return Top.step;
}
Node next;
//rotates:
next.x = Top.x; next.y = Top.y; next.str = g[next.x][next.y]; next.cnt = Top.cnt + ; next.step = Top.step + ; next.f = Get_F(next, e.x, e.y);
if (vis[next.x][next.y][next.cnt % ] == )
{
vis[next.x][next.y][next.cnt % ] = ;
Q.push(next);
} //Move:
next.cnt = Top.cnt; next.str = g[next.x][next.y];
char op = rotates(next.str, next.cnt);
for (int i = ; i<Move[op].size(); i++)
{
next.x = Top.x + dist[Move[op][i]][];
next.y = Top.y + dist[Move[op][i]][];
if (check(next.x, next.y) && g[next.x][next.y] != '*'&&!vis[next.x][next.y][next.cnt % ])
{
next.str = g[next.x][next.y];
if (checkMove(next, Top,Move[op][i]))//检测是否B也能到A
{
vis[next.x][next.y][next.cnt % ] = ;
next.step = Top.step + ;
next.f = Get_F(next, e.x, e.y);
Q.push(next);
}
}
}
}
return -;
}
int main()
{
//#ifdef CYM
// freopen("in.txt", "r", stdin);
// freopen("out.txt", "w", stdout);
//#endif
init();
while (~scanf("%d%d", &n, &m)){
for (int i = ; i<n; i++)
{
scanf("%s", g[i]);
}
Node start, end;
scanf("%d%d", &start.x, &start.y);
start.x--; start.y--;
scanf("%d%d", &end.x, &end.y);
end.x--; end.y--;
start.step = , start.str = g[start.x][start.y], start.cnt = ; start.f = Get_F(start, end.x, end.y);
printf("%d\n", dfs(start, end));
}
return ;
}
Codeforces Round #354 (Div. 2)-D的更多相关文章
- Codeforces Round #354 (Div. 2) ABCD
Codeforces Round #354 (Div. 2) Problems # Name A Nicholas and Permutation standard input/out ...
- Codeforces Round #354 (Div. 2)-C
C. Vasya and String 题目链接:http://codeforces.com/contest/676/problem/C High school student Vasya got a ...
- Codeforces Round #354 (Div. 2)-B
B. Pyramid of Glasses 题目链接:http://codeforces.com/contest/676/problem/B Mary has just graduated from ...
- Codeforces Round #354 (Div. 2)-A
A. Nicholas and Permutation 题目链接:http://codeforces.com/contest/676/problem/A Nicholas has an array a ...
- Codeforces Round #354 (Div. 2) D. Theseus and labyrinth
题目链接: http://codeforces.com/contest/676/problem/D 题意: 如果两个相邻的格子都有对应朝向的门,则可以从一个格子到另一个格子,给你初始坐标xt,yt,终 ...
- Codeforces Round #354 (Div. 2) C. Vasya and String
题目链接: http://codeforces.com/contest/676/problem/C 题解: 把连续的一段压缩成一个数,对新的数组求前缀和,用两个指针从左到右线性扫一遍. 一段值改变一部 ...
- Codeforces Round #354 (Div. 2)_Vasya and String(尺取法)
题目连接:http://codeforces.com/contest/676/problem/C 题意:一串字符串,最多改变k次,求最大的相同子串 题解:很明显直接尺取法 #include<cs ...
- Codeforces Round #354 (Div. 2) E. The Last Fight Between Human and AI 数学
E. The Last Fight Between Human and AI 题目连接: http://codeforces.com/contest/676/problem/E Description ...
- Codeforces Round #354 (Div. 2) D. Theseus and labyrinth bfs
D. Theseus and labyrinth 题目连接: http://www.codeforces.com/contest/676/problem/D Description Theseus h ...
随机推荐
- Mac shell 添加VPN 路由
苦于每次使用内网VPN,都需要手动添加route来设置网络的映射,才可以正常使用VPN. 每次都需要自动添加,当然就是觉得很烦,想到Mac的shell.反正每次添加的时候都是在终端执行添加路由.为了方 ...
- Spring@Autowired注解与自动装配
1 配置文件的方法 我们编写spring 框架的代码时候.一直遵循是这样一个规则:所有在spring中注入的bean 都建议定义成私有的域变量.并且要配套写上 get 和 set方法. Boss ...
- vector在C++中的基本用法
在写BlackJackGame的时候,考虑到要用到容器,所以就对容器的相关知识强化了一下: 因为我想的是有card类,最后要实现发牌,洗牌等等一系列的操作的时候,使用指向card类的对象的指针,将ca ...
- Java常用工具类题库
一. 填空题 在Java中每个Java基本类型在java.lang包中都在一个相应的包装类,把基本类型数据转换为对象,其中包装类Integer是___Number__的直接子类. 包装类Inte ...
- objective-c可变数组
1 #pragma mark ---------------可变数组----------------- 2 // 可以在数组里面进行增删改的操作 3 // 4 // ...
- Oracle读写分离架构
读写分离是架构分布式系统的一个重要思想.不少系统整体处理能力并不能同业务的增长保持同步,因此势必会带来瓶颈,单纯的升级硬件并不能一劳永逸.针对业务类型特点,需要从架构模式上进行一系列的调整,比如业务模 ...
- Android ArrayAdapter使用
1. 可以直接使用getContext()获取Context对象 2. 可以在构造方法中传入context, 数据对象的列表, super(context, 0, object);完成Adapter的 ...
- EmguCV学习 与opencv的区别和联系
openCV是因特尔的一个开源的视觉库,里面几乎包含了所有的图像处理的经典算法,并且采用C和少量的C++编写,运行效率很高,对于做图像处理这方面工作的,认识opencv是必须的工作.不过opencv有 ...
- HTML学习之Web存储(五)
本地数据库功能大大增强了Web应用对于本地存储数据的方式和功能.Web时代真正进入了:“客户端为重,服务端为轻的时代”. <!DOCTYPE html> <html xmlns=&q ...
- Delphi中的函数指针判断是否为空
delphi函数指针 只有@@p才代表了函数指针本身的地址 assigned(p) 判断是否为空 或者用 @p=nil 来判断函数指针是不是为空 Delphi中的函数指针实际上就是指针,只是在使用 ...