【NOIP2013提高组】华容道
分析
一个比较显然的方式是
设 \(f_{i,j,x,y}\) 表示达到空格所处位置为 \((i,j)\) 且特殊格位置为 \(x,y\) 的状态的最少步数
一次可以交换空格和相邻格,代价为 \(1\),\(bfs\) 转移即可
但确实时间无法接受
我们想到转移时
当且仅当空格和特殊格相邻时特殊格的位置才可能变
所以我们设 \(f_{i,j,k}\) 表示特殊格位置为 \((i,j)\) 且空格在特殊格 \(k(k\in[0,3])\) 方向的最小步数
那么考虑两种转移
1.不动特殊格,空格从 \(k\) 方向转到 \(l\) 方向
2.空格与特殊格交换
第一种可以 \(bfs\) 预处理出来,记为 \(g_{i,j,k,l}\)
第二种代价为 \(1\)
因为有两种代价,所以我们要 \(spfa\) 转移
交换空格会使空格所在特殊格的方向相反
为方便表示我们用 \({1,2,3,4}\) 表示上下左右
\(Code\)
#include<cstdio>
#include<iostream>
#include<cstring>
using namespace std;
const int N = 35, INF = 0x3f3f3f3f;
int n, m, q, a[N][N], f[N][N][5], g[N][N][5][5], dis[N][N], vis[N][N][5];
int fx[4][2] = {{-1, 0}, {1, 0}, {0, -1}, {0, 1}};
struct node{int x, y, k;}Q[N*N*N];
inline int judge(int x, int y){return (x >= 1 && x <= n && y >= 1 && y <= m && a[x][y]);}
inline int bfs(int sx, int sy, int tx, int ty)
{
memset(dis, 0x3f3f3f3f, sizeof dis);
if (!judge(sx, sy) || !judge(tx, ty)) return INF;
dis[sx][sy] = 0;
int head = 0, tail = 1;
Q[1] = node{sx, sy, 0};
while (head < tail)
{
node now = Q[++head];
if (dis[tx][ty] != INF) return dis[tx][ty];
for(register int k = 0; k < 4; k++)
{
int x = now.x + fx[k][0], y = now.y + fx[k][1];
if (judge(x, y) && dis[x][y] > dis[now.x][now.y] + 1)
dis[x][y] = dis[now.x][now.y] + 1, Q[++tail] = node{x, y, 0};
}
}
return dis[tx][ty];
}
void prepare()
{
for(register int i = 1; i <= n; i++)
for(register int j = 1; j <= m; j++)
{
int tmp = a[i][j];
a[i][j] = 0;
for(register int k = 0; k < 4; k++)
for(register int l = 0; l < 4; l++)
g[i][j][k][l] = bfs(i + fx[k][0], j + fx[k][1], i + fx[l][0], j + fx[l][1]);
a[i][j] = tmp;
}
}
inline int spfa(int sx, int sy, int tx, int ty)
{
memset(vis, 0, sizeof vis);
int head = 0, tail = 0;
for(register int k = 0; k < 4; k++)
{
int x = sx + fx[k][0], y = sy + fx[k][1];
if (judge(x, y)) Q[++tail] = node{sx, sy, k}, vis[sx][sy][k] = 1;
}
while (head < tail)
{
node now = Q[++head];
for(register int k = 0; k < 4; k++)
{
int x = now.x + fx[k][0], y = now.y + fx[k][1];
if (judge(x, y) && f[x][y][k ^ 1] > f[now.x][now.y][now.k] + g[now.x][now.y][now.k][k] + 1)
{
f[x][y][k ^ 1] = f[now.x][now.y][now.k] + g[now.x][now.y][now.k][k] + 1;
if (!vis[x][y][k ^ 1]) Q[++tail] = node{x, y, k ^ 1}, vis[x][y][k ^ 1] = 1;
}
}
vis[now.x][now.y][now.k] = 0;
}
int ans = INF;
for(register int k = 0; k < 4; k++) ans = min(ans, f[tx][ty][k]);
return (ans == INF ? -1 : ans);
}
inline int solve()
{
int ex, ey, sx, sy, tx, ty;
scanf("%d%d%d%d%d%d", &ex, &ey, &sx, &sy, &tx, &ty);
if (!judge(sx, sy) || !judge(tx, ty)) return -1;
if (sx == tx && sy == ty) return 0;
memset(f, 0x3f3f3f3f, sizeof f);
int tmp = a[sx][sy];
a[sx][sy] = 0;
for(register int k = 0; k < 4; k++)
f[sx][sy][k] = bfs(ex, ey, sx + fx[k][0], sy + fx[k][1]);
a[sx][sy] = tmp;
return spfa(sx, sy, tx, ty);
}
int main()
{
scanf("%d%d%d", &n, &m, &q);
for(register int i = 1; i <= n; i++)
for(register int j = 1; j <= m; j++) scanf("%d", &a[i][j]);
prepare();
for(register int i = 1; i <= q; i++) printf("%d\n", solve());
}
【NOIP2013提高组】华容道的更多相关文章
- [NOIP2013 提高组] 华容道 P1979 洛谷
[NOIP2013 提高组] 华容道 P1979 洛谷 强烈推荐,更好的阅读体验 经典题目:spfa+bfs+转化 题目大意: 给出一个01网格图,和点坐标x,y空格坐标a,b,目标位置tx,ty要求 ...
- [NOIP2013提高组]华容道
这道题第一眼看是暴力,然后发现直接暴力会TLE. 把问题转换一下:移动空格到处跑,如果空格跑到指定位置的棋子,交换位置. 这个可以设计一个状态:$[x1][y1][x2][y2]$,表示空格在$(x1 ...
- [NOIp2013提高组]积木大赛/[NOIp2018提高组]铺设道路
[NOIp2013提高组]积木大赛/[NOIp2018提高组]铺设道路 题目大意: 对于长度为\(n(n\le10^5)\)的非负数列\(A\),每次可以选取一个区间\(-1\).问将数列清零至少需要 ...
- [NOIP2013提高组] CODEVS 3287 火车运输(MST+LCA)
一开始觉得是网络流..仔细一看应该是最短路,再看数据范围..呵呵不会写...这道题是最大生成树+最近公共祖先.第一次写..表示各种乱.. 因为要求运输货物质量最大,所以路径一定是在最大生成树上的.然后 ...
- 【NOIP2013提高组T3】加分二叉树
题目描述 设一个n个节点的二叉树tree的中序遍历为(1,2,3,…,n),其中数字1,2,3,…,n为节点编号.每个节点都有一个分数(均为正整数),记第i个节点的分数为di,tree及它的每个子树都 ...
- NOIP2013 提高组day2 3 华容道 BFS
描述 小 B 最近迷上了华容道,可是他总是要花很长的时间才能完成一次.于是,他想到用编程来完成华容道:给定一种局面,华容道是否根本就无法完成,如果能完成,最少需要多少时间. 小 B 玩的华容道与经典的 ...
- [NOIP2013] 提高组 洛谷P1979 华容道
题目描述 [问题描述] 小 B 最近迷上了华容道,可是他总是要花很长的时间才能完成一次.于是,他想到用编程来完成华容道:给定一种局面, 华容道是否根本就无法完成,如果能完成, 最少需要多少时间. 小 ...
- 洛谷P1979 [NOIP2013提高组Day2T3]华容道
P1979 华容道 题目描述 [问题描述] 小 B 最近迷上了华容道,可是他总是要花很长的时间才能完成一次.于是,他想到用编程来完成华容道:给定一种局面, 华容道是否根本就无法完成,如果能完成, 最少 ...
- 3537. 【NOIP2013提高组day2】华容道(搜索 + 剪枝)
Problem 给出一个类似华容道的图.\(q\)次询问,每次给你起始点,终止点,空格位置,让你求最少步数 \(n,m\le 30, q\le 500\). Soultion 一道智障搜索题. 弱智想 ...
- NOIP2013提高组D2T3 华容道
n<=30 * m<=30 的地图上,0表示墙壁,1表示可以放箱子的空地.q<=500次询问,每次问:当空地上唯一没有放箱子的空格子在(ex,ey)时,把位于(sx,sy)的箱子移动 ...
随机推荐
- orcl substr函数与java substring 的不同
前天事情急改一个存储过程时遇到了substr方法时,一直用好,然后用其他方法跳过去了,今天有时间回头来验证 才发现和java太不一样了! select substr('为中华之崛起而读书',2,4) ...
- 2.10:数据加工与展示-pandas清洗、Matplotlib绘制
〇.目标 1. 使用pandas完成基本的数据清洗加工处理: 2. 使用Matplotlib进行简单的数据图形化展示. 一.用pandas清洗处理数据 1.判断是否存在空值 数据缺失在很多数据中存在, ...
- redis集合 实现 队列
先说一下需求:用队列解决 流量削峰,主要应用场景:商城秒杀功能. 以下是业务流程图可以参考一下: 然后本地实现思路 截图下单页面 每次购买数量会减少1,设置了1000个库存,用户id 是随机生成的. ...
- 聚焦技术,锐意创新,GaussDB给世界一个更优选择
摘要:从整个行业应用层面来看,现在,数据库的国产化时代已经到来. 本文分享自华为云社区<聚焦技术,锐意创新,GaussDB给世界一个更优选择>,作者: GaussDB数据库. 今天,以&q ...
- STL set容器常用API
set容器,容器内部将数据自动排序(平衡二叉树),不能插入重复元素.multiset可以插入重复元素.不能修改容器中的值,通过删除值,在插入. #define _CRT_SECURE_NO_WARNI ...
- [OpenCV实战]22 使用EigenFaces进行人脸重建
目录 1 背景 1.1 什么是EigenFaces? 1.2 坐标的变化 2 面部重建 2.1 计算新面部图像的PCA权重 2.2 使用EigenFaces进行面部重建 3 参考 在这篇文章中,我们将 ...
- [python] python-pinyin库
python-pinyin库是一个汉字拼音转换工具,其主要功能有: 根据词组智能匹配最正确的拼音. 支持多音字. 简单的繁体支持, 注音支持. 支持多种不同拼音风格. 安装命令为:pip instal ...
- 统一返回对象封装和统一异常捕获封装springboot starter
好久没有更新文章了,高龄开发没什么技术,去了外包公司后没怎么更新文章了.今天分享下统一处理starter,相信开发web系统的时候都是会涉及到前后端的交互,而后端返回数据的时候一般都会统一封装一个返回 ...
- 图文并茂quasar2.6+vue3+ts+vite创建项目并引入mockjs,mockjs 拦截ajax请求的原理是什么,quasar为什么要使用boot?
每天都要开心(▽)哇: 首先呢,我们来创建项目 执行下面命令,开始创建项目啦 $ npm i -g @quasar/cli $ npm init quasar 下面是我的选项,仅供参考哇 √ What ...
- 《深入理解Java虚拟机》第三章读书笔记(一)——垃圾回收算法
参考书籍<深入理解java虚拟机>周志明著 系列文章目录和关于我 本文主要介绍垃圾回收理论知识 1.jvm哪些区域需要进行垃圾回收 虚拟机栈,本地方法栈,程序计数器都是线程私有的,随线程而 ...