3537. 【NOIP2013提高组day2】华容道(搜索 + 剪枝)

Problem

给出一个类似华容道的图。\(q\)次询问,每次给你起始点,终止点,空格位置,让你求最少步数

\(n,m\le 30, q\le 500\).

Soultion

一道智障搜索题。

弱智想法最多80分。不用想了。我已经试过所有非O2的常数优化,还是有1.05秒。

考虑一下预处理。

事实上,我们发现只有当空格位置在初始点旁边时才会影响初始点(废话),所以我们可以先预处理在某一个点\((x,y)\)的四周,不经过这个点(x,y),到达这个点四周的最少步数。BFS解决。

然后每次询问时,就先把空格位置跑到初始点的四周,然后每次用预处理的去更新。

Code
#include <bits/stdc++.h>

#define I register int
#define F(i, a, b) for (I i = a; i <= b; i ++)
#define mem(a, b) memset(a, b, sizeof a)

const int N = 31;

const int dx[4] = { 1, 0 , 0, - 1};
const int dy[4] = { 0, - 1 , 1, 0};

using namespace std;

int h, t, Ans, ex, ey, sx, sy, tx, ty;
int n, m, q, a[N + 1][N + 1];
int f[N][N][4], bz[N][N][N][N], vis[N][N];
struct node {
	int x, y, k;
} d[N * N * N];
struct Node {
	int x, y;
} D[N * N];

void Doit() {
	int st = 0, en = 0;

	mem(f, 7), h = 0, D[t = 1] = {ex, ey}, mem(vis, 0), vis[ex][ey] = 1;
	F(p, 0, 3) {
		I xxx = ex + dx[p], yyy = ey + dy[p];
		if (a[xxx][yyy] && xxx == sx && yyy == sy)
			d[++ en] = {ex, ey, 3 - p}, f[ex][ey][3 - p] = 1;
	}
	while (h ++ < t) {
		I x = D[h].x, y = D[h].y;
		F(k, 0, 3) {
			I xx = x + dx[k], yy = y + dy[k];
			if (a[xx][yy] && !vis[xx][yy] && !(xx == sx && yy == sy)) {
				vis[xx][yy] = vis[x][y] + 1, D[++ t] = {xx, yy};
				F(p, 0, 3) {
					I xxx = xx + dx[p], yyy = yy + dy[p];
					if (a[xxx][yyy] && xxx == sx && yyy == sy)
						d[++ en] = {xx, yy, 3 - p}, f[xx][yy][3 - p] = vis[xx][yy];
				}
			}
		}
	}

	while (st ++ < en) {
		I x = d[st].x, y = d[st].y, k = d[st].k;
		F(p, 0, 3) {
			I xx = x + dx[p], yy = y + dy[p];
			if (a[xx][yy] && f[xx][yy][p] > f[x][y][k] + bz[x][y][k][3 - p]) {
				f[xx][yy][p] = f[x][y][k] + bz[x][y][k][3 - p];
				d[++ en] = {xx, yy, p};
			}
		}
	}

	int Ans = 1e8;
	F(k, 0, 3)
		Ans = min(Ans, f[tx][ty][k]);
	printf("%d\n", Ans == 1e8 ? - 1 : Ans);
}

int main() {	
	scanf("%d%d%d", &n, &m, &q);
	F(i, 1, n)
		F(j, 1, m)
			scanf("%d", &a[i][j]);

	mem(bz, 7);
	F(i, 1, n)
		F(j, 1, m) {
			if (!a[i][j]) continue;
			F(k, 0, 3) {
				int x = i + dx[k], y = j + dy[k], w = k;
				if (!a[x][y]) continue;
				h = 0, D[t = 1] = {x, y}, mem(vis, 0), bz[i][j][w][w] = 1, vis[x][y] = 1;
				while (h ++ < t) {
					x = D[h].x, y = D[h].y;
					F(k, 0, 3) {
						int xx = x + dx[k], yy = y + dy[k];
						if (a[xx][yy] && !(xx == i && yy == j) && !vis[xx][yy]) {
							vis[xx][yy] = vis[x][y] + 1, D[++ t] = {xx, yy};
							F(k, 0, 3) {
								int xxx = xx + dx[k], yyy = yy + dy[k];
								if (a[xxx][yyy] && xxx == i && yyy == j)
									bz[i][j][3 - w][k] = vis[xx][yy];
							}
						}
					}
				}
			}
		}

	F(i, 1, q) {
		scanf("%d%d%d%d%d%d", &ex, &ey, &sx, &sy, &tx, &ty);
		if (sx == tx && sy == ty) {
			puts("0");
			continue;
		}
		Doit();
	}
}
posted @ 2019-04-10 21:39  proking  阅读(195)  评论(0编辑  收藏  举报