A Dicey Problem 骰子难题(Uva 810)
题目描述:https://uva.onlinejudge.org/external/8/810.pdf
把一个骰子放在一个M x N的地图上,让他按照规定滚动,求滚回原点的最短路径。
思路: 记忆化搜索(我这里用的dfs深度优先搜索)
难点:
如何推导骰子的状态
我没有用打表,而是先用build_dice()初始化二维数组dice, 这样,当我们有骰子的顶面top,正面face, 就可以通过dice[top][face] 获得对应的右面,由此,就可以实现骰子的旋转。rotate()函数实现骰子的旋转,给它top, face, 和旋转的方向(前,后,左,右), 它会返回旋转后骰子的top, face.
记忆化搜索要多一个状态记录骰子的状态
说白了就是记录访问状态的数组vis要多两个维度分别记录top, face,因为top, face一旦固定,骰子就固定了
Note: 输出的格式也要稍微注意一下,锁进、末尾的逗号,换行要处理好。
#include <iostream>
#include <cstdio>
#include <vector>
#include <utility>
#include <cstring>
using namespace std;
const int MAXN = + ;
int b[MAXN][MAXN], R, C, sx, sy, st, sf, len;
int dx[] = {, -, , }, dy[] = {, , -, }, dice[][];
bool vis[MAXN][MAXN][][], ok, first;
vector<pair<int, int> > path;
void build_dice() { //dice[top][face] = right
dice[][] = ;
for (int i = ; i < ; i++) {
for (int j = ; j < ; j++) {
if (i == j) continue;
if (dice[i][ - j]) dice[i][j] = - dice[i][ - j];
if (dice[i][j]) {
dice[j][i] = - dice[i][j];
dice[i][dice[i][j]] = - j;
}
}
}
}
pair<int, int> rotate(int top, int face, int di) {
pair<int, int> n_tf;
if (di == ) {
n_tf.second = face;
n_tf.first = - dice[top][face];
}
else if (di == ) {
n_tf.first = face;
n_tf.second = - top;
}
else if (di == ) {
n_tf.second = face;
n_tf.first = dice[top][face];
}
else {
n_tf.second = top;
n_tf.first = - face;
}
return n_tf;
}
bool inside(int x, int y) {
return x > && y > && x <= R && y <= C;
}
void dfs(int x, int y, int t, int f) {
path.push_back(make_pair(x, y));
if (x == sx && y == sy)
if (first) first = ;
else { ok = ; return;}
vis[x][y][t][f] = ;
for (int i = ; i < ; i++) {
pair<int, int> nextp = rotate(t, f, i);
int nx = x + dx[i], ny = y + dy[i];
int nt = nextp.first, nf = nextp.second;
if (inside(nx, ny) && (!vis[nx][ny][nt][nf] || (nx == sx && ny == sy)) && (t == b[nx][ny] || b[nx][ny] == -)) {
dfs(nx, ny, nt, nf);
}
if (ok) return;
}
vis[x][y][t][f] = ;
path.pop_back();
}
int main() {
build_dice();
char name[];
while (scanf("%s", name) == && strcmp(name, "END")) {
scanf("%d%d%d%d%d%d", &R, &C, &sx, &sy, &st, &sf);
memset(vis, , sizeof(vis));
memset(b, , sizeof(b));
path.clear(); ok = ; first = ;
for (int i = ; i <= R; i++)
for (int j = ; j <= C; j++)
scanf("%d", &b[i][j]);
dfs(sx, sy, st, sf);
len = path.size();
printf("%s", name);
if (ok)
for (int i = ; i < len; i++) {
if (i % == ) {
if (i >= ) printf(",");
printf("\n (%d,%d)", path[i].first, path[i].second);
}
else printf(",(%d,%d)", path[i].first, path[i].second);
}
else printf("\n No Solution Possible");
printf("\n");
}
return ;
}
A Dicey Problem 骰子难题(Uva 810)的更多相关文章
- UVA 810 - A Dicey Problem(BFS)
UVA 810 - A Dicey Problem 题目链接 题意:一个骰子,给你顶面和前面.在一个起点,每次能移动到周围4格,为-1,或顶面和该位置数字一样,那么问题来了,骰子能不能走一圈回到原地, ...
- poj 1872 A Dicey Problem WA的代码,望各位指教!!!
A Dicey Problem Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 832 Accepted: 278 Des ...
- UVA-810 A Dicey Problem (BFS)
题目大意:滚骰子游戏,骰子的上面的点数跟方格中的数相同时或格子中的数是-1时能把格子滚过去,找一条从起点滚到起点的路径. 题目大意:简单BFS,状态转移时细心一些即可. 代码如下; # include ...
- poj1872A Dicey Problem
Home Problems Status Contest 284:28:39 307:00:00 Overview Problem Status Rank A B C D E F G H ...
- Uva - 810 - A Dicey Problem
根据状态进行bfs,手动打表维护骰子滚动. AC代码: #include <iostream> #include <cstdio> #include <cstdlib&g ...
- UVA 810 A Dicey Promblem 筛子难题 (暴力BFS+状态处理)
读懂题意以后还很容易做的, 和AbbottsRevenge类似加一个维度,筛子的形态,可以用上方的点数u和前面的点数f来表示,相对的面点数之和为7,可以预先存储u和f的对应右边的点数,点数转化就很容易 ...
- UVA 11991 Easy Problem from Rujia Liu?【STL】
题目链接: option=com_onlinejudge&Itemid=8&page=show_problem&problem=3142">https://uv ...
- UVa 11375 - Matches
http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&p ...
- UVA 567 Risk【floyd】
题目链接: option=com_onlinejudge&Itemid=8&page=show_problem&problem=508">https://uva ...
随机推荐
- Laravel_1 安装
1>http://www.golaravel.com/post/install-and-run-laravel-5-x-on-windows/ 2>http://www.golaravel ...
- js小分享
之前实现一些js代码时,总觉得无法下手,所以最近在学习一下特别细的知识点,分享笔记.嘻嘻,偷个小懒,我把自己的笔记拍个照片就不打字了.嘎嘎,放心放心,自觉得字写的还算ok的啦- 表示家里的老弟玩游戏, ...
- java_泛型(2016-11-17)
没有自己敲,这篇博客讲的不错,直接记录. 犯懒啊 重点关注 T,?以及擦除 Java总结篇系列:Java泛型:http://www.cnblogs.com/lwbqqyumidi/p/3837629. ...
- 制作Net程序的帮助文档--总结
一.工具的准备 目前,一般采用Sandcastle Help File Builder工具来制作.Net程序帮助文档,该工具主要是利用Xml文档里的信息以及DLL文件来生成完整的帮助文档.在Visua ...
- Quartz.NET开源作业调度架构
Quartz.NET是一个开源的作业调度框架,是 OpenSymphony 的 Quartz API 的.NET移植,它用C#写成,可用于winform和asp.net应用中.它提供了巨大的灵活性而不 ...
- JavaScript学习总结【3】、JS对象
在 JS 中一切皆对象,并提供了多个内置对象,比如:String.Array.Date 等,此外还支持自定义对象.对象只是一种特殊类型的数据,并拥有属性和方法,属性是与对象相关的值,方法是能够在对象上 ...
- runas /user:administrator cmd 以管理员身份运行CMD
runas /user:administrator cmd 以管理员身份运行CMD 1.windows+r打开 2.然后根据提示输入密码
- js获取时间天数
date2必须大于date1 function getDays(date1,date2){ /*获取之间的天数*/ /*date1,date2都是date格式*/ var getd=(date2.ge ...
- Java控制台版推箱子
import java.util.Scanner; public class b { public static void main(String[] args) { Scanner input = ...
- str、__str__ 、repr、 __repr__、``
内建函数str()和repr() 或反引号操作符(``)可以方便地以字符串的方式获取对象的内容.类型.数值属性等信息. str()函数得到的字符串可读性好(故被print调用),而结果通常无法用eva ...