【题解】APIO2013机器人
其实这题前前后后的思考时间加起来应该有两天之久了,dp状态,转移方式等等都还是比较好想,然而左看右看觉得spfa复杂度未免太爆炸……然后选择看了一篇题解,发现在多重优化之下,其实是可以过的……
首先建立状态,这个应该比较明显:\(f[l][r][x][y]\) 代表合并完区间 \(l\) ~\(r\) 之后,机器人停在 \(x,y\) 处所需要的最少移动次数。转移状态即为:
\(f[l][r][x][y] = f[l][k][x][y] + f[k + 1][r][x][y] \left ( l <= k <= r \right )\)
\(f[l][r][x][y] = f[l][r][x'][y'] + 1 \)
其中第二个转移发生的条件是 \(x',y'\) 可以一步到达 \(x,y\)。第二个转移状态就是在之前的博客中所提及的那样:1.满足三角形不等式;2.不满足拓扑序;针对这样的转移,我们用 spfa 来优化 dp 的转移。注意在这张图中,边权均为1。在单源的最短路中,这样的图spfa可以优化为bfs, 在多源最短路中我们可以使用两个队列来进行优化。这两个队列分别存储新增的节点 & 被松弛所以要去松弛其余节点的节点。这样将节点分类之后,每一次取出队首元素权值更小的进行松弛操作。我们会发现第二个队列中节点的权值是单调的(在边权为1的图中,先访问到的节点权值更小),而第一个队列中的元素我们使用基数排序来排。(并不知道为什么要用基数排序,或许就是比较快吧?)
然后这份代码是我抄的大佬的代码,非常感谢了。其中有一个小小的技巧:memset的时候默认赋给节点当前数据类型的最大值,相加会溢出。但对于这种没有正负要求的,我们可以利用 unsigned 自然溢出使得结果依然是最大值。(・ω<)☆ 感觉这题还是挺毒的,差点就被毒死了……
#include <bits/stdc++.h>
using namespace std;
#define maxn 505
#define maxk 400000
#define uns unsigned short
#define INF 32639
int n, W, H, ans = INF;
int mark[maxn][maxn][];
int L, R;
uns f[][][maxn][maxn];
int cnt, top, tank[maxk], S[maxk];
char Map[maxn][maxn];
bool vis[maxn][maxn];
int dxy[][] = {{,}, {,}, {,-}, {-,}}; struct node
{
int x, y;
node(int xx = , int yy = ) { x = xx, y = yy; }
}g[maxn][maxn][], pos[], q[maxk];
queue <node> q1, q2; int read()
{
int x = , k = ;
char c;
c = getchar();
while(c < '' || c > '') { if(c == '-') k = -; c = getchar(); }
while(c >= '' && c <= '') x = x * + c - '', c = getchar();
return x * k;
} void gmin(uns &x, uns y) { x = x > y ? y : x; } node dfs(int x, int y, int k)
{
if(mark[x][y][k] == cnt) return g[x][y][k] = node (-, -);
mark[x][y][k] = cnt;
if(g[x][y][k].x != && g[x][y][k].y != ) return g[x][y][k];
int pre = k;
if(Map[x][y] == 'A') k = (k + ) % ;
else if(Map[x][y] == 'C') k = (k + ) % ;
int xx = x + dxy[k][], yy = y + dxy[k][];
if(xx < || yy < || xx > H || yy > W || Map[xx][yy] == 'x') return g[x][y][pre] = node(x, y);
return g[x][y][pre] = dfs(xx, yy, k);
} void spfa()
{
memset(tank, , sizeof(tank));
for(int i = ; i <= top; i ++) tank[f[L][R][q[i].x][q[i].y]] ++;
for(int i = ; i <= INF; i ++) tank[i] += tank[i - ];
for(int i = ; i <= top; i ++) S[tank[f[L][R][q[i].x][q[i].y]] --] = i;
for(int i = ; i <= top; i ++) q1.push(q[S[i]]);
top = ;
while(!q1.empty() || !q2.empty())
{
node now;
if(q1.empty()) now = q2.front(), q2.pop();
else if(q2.empty()) now = q1.front(), q1.pop();
else
{
int x1 = q1.front().x, y1 = q1.front().y;
int x2 = q2.front().x, y2 = q2.front().y;
if(f[L][R][x1][y1] <= f[L][R][x2][y2]) now = q1.front(), q1.pop();
else now = q2.front(), q2.pop();
}
vis[now.x][now.y] = ;
for(int i = ; i < ; i ++)
{
node v = g[now.x][now.y][i];
if(v.x == - || v.y == -) continue;
if(f[L][R][v.x][v.y] > f[L][R][now.x][now.y] + )
{
f[L][R][v.x][v.y] = f[L][R][now.x][now.y] + ;
if(!vis[v.x][v.y]) vis[v.x][v.y] = , q2.push(v);
}
}
}
} int main()
{
n = read(), W = read(), H = read();
memset(f, , sizeof(f));
for(int i = ; i <= H; i ++)
{
scanf("%s", Map[i] + );
for(int j = ; j <= W; j ++)
if(Map[i][j] > '' && Map[i][j] <= '')
pos[Map[i][j] - ''] = node(i, j);
}
for(int i = ; i <= H; i ++)
for(int j = ; j <= W; j ++)
if(Map[i][j] != 'x')
for(int k = ; k < ; k ++)
++ cnt, dfs(i, j, k);
for(int i = ; i <= n; i ++)
{
vis[pos[i].x][pos[i].y] = ;
q[++ top] = pos[i];
L = R = i; f[i][i][pos[i].x][pos[i].y] = ;
spfa();
}
for(int l = , j; l <= n; l ++)
for(int i = ; (j = i + l - ) <= n; i ++)
{
for(int x = ; x <= H; x ++)
for(int y = ; y <= W; y ++)
{
for(int k = i; k < j; k ++)
gmin(f[i][j][x][y], f[i][k][x][y] + f[k + ][j][x][y]);
if(f[i][j][x][y] < INF) q[++ top] = node(x, y), vis[x][y] = ;
}
L = i, R = j; spfa();
}
unsigned short ans = INF;
for(int i = ; i <= H; i ++)
for(int j = ; j <= W; j ++)
ans = min(ans, f[][n][i][j]);
if(ans < INF) printf("%u\n", ans);
else printf("-1\n");
return ;
}
【题解】APIO2013机器人的更多相关文章
- bzoj3205 [Apio2013]机器人
3205: [Apio2013]机器人 Time Limit: 15 Sec Memory Limit: 128 MBSubmit: 953 Solved: 227[Submit][Status] ...
- [Bzoj3205][Apio2013]机器人(斯坦纳树)(bfs)
3205: [Apio2013]机器人 Time Limit: 15 Sec Memory Limit: 128 MBSubmit: 977 Solved: 230[Submit][Status] ...
- [APIO2013]机器人(斯坦纳树)
题目描述 VRI(Voltron 机器人学会)的工程师建造了 n 个机器人.任意两个兼容的机 器人站在同一个格子时可以合并为一个复合机器人. 我们把机器人用 1 至 n 编号(n ≤ 9).如果两个机 ...
- bzoj千题计划230:bzoj3205: [Apio2013]机器人
http://www.lydsy.com/JudgeOnline/problem.php?id=3205 历时一天,老子终于把它A了 哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈 因为不懂spfa ...
- [APIO2013]机器人[搜索、斯坦纳树]
题意 题目链接 分析 记 g(d,x,y) 表示从 (x,y) 出发,方向为 d 到达的点,这个可以通过记忆化搜索求出,注意如果转移成环(此时向这个方向走没有意义)要特判. 记 f(l,r,x,y) ...
- BZOJ3205/UOJ107 [Apio2013]机器人
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...
- [APIO2013]机器人
题目描述 VRI(Voltron 机器人学会)的工程师建造了 n 个机器人.任意两个兼容的机 器人站在同一个格子时可以合并为一个复合机器人. 我们把机器人用 1 至 n 编号(n ≤ 9).如果两个机 ...
- bzoj 3205: [Apio2013]机器人【dfs+斯坦纳树+spfa】
第一次听说斯坦纳树这种东西 先dfs预处理出来dis[i][j][k]表示格子(i,j)向k方向转移能到哪,记忆话搜索预处理,注意如果有环的话特判一下 设f[i][j][x][y]表示复合机器人i-j ...
- 【BZOJ3205_洛谷3638】[APIO2013]机器人(动态规划)
题目: 洛谷3638 分析: 卡了一天的神题--(OrzJumpmelon) 首先预处理出从点\(p\)向\(d\)方向出发最终能到达的点\(nxt[p][d]\).这个可以直接记忆化搜索解决.如果出 ...
随机推荐
- python的字典数据类型及常用操作
字典的定义与特性 字典是Python语言中唯一的映射类型. 定义:{key1: value1, key2: value2} 1.键与值用冒号“:”分开: 2.项与项用逗号“,”分开: 特性: 1.ke ...
- python3 练习题100例 (二十九)猴子吃桃问题
题目内容: 猴子第一天摘下若干个桃子,当即吃了一半,还不过瘾,又多吃了一个第二天早上又将剩下的桃子吃掉一半,又多吃了一个.以后每天早上都吃了前一天剩下的一半零一个.到第n天(<1<n< ...
- Scrapy核心组件
• 引擎(Scrapy)用来处理整个系统的数据流处理, 触发事务(框架核心) • 调度器(Scheduler)用来接受引擎发过来的请求, 压入队列中, 并在引擎再次请求的时候返回. 可以想像成一个UR ...
- git的基本操作总结
参考链接 https://blog.csdn.net/u012661010/article/details/73433872 https://blog.csdn.net/shj_php/article ...
- python中矢量化字符串方法
- shell重温---基础篇(参数传递&echo命令)
经过前两天的学习,关于shell的基础算是知道的一般般啦,最起码不算是小白了(纯属意淫).今天就来点干货哈. 首先是运行shell脚本时的参数传递.脚本内获取参数的格式为$n.n代表了一个数字,例 ...
- ChemDraw Std 14性价比最高版本,即将下架
虽然ChemDraw Std 14是ChemOffice®14的基础组件,但是基础功能涵盖全面,是教育专供产品.根据官方最新消息ChemDraw系列软件产品线将进行全面的升级,ChemOffice®1 ...
- 当我们访问不了虚拟机上ip上的web页面,是因为在window上要添加映射
在主机上添加映射步骤 1.打开C盘 注意:用nopedata++打开 保存即可!
- Wireshark lua dissector 对TCP消息包合并分析
应用程序发送的数据报都是流式的,IP不保证同一个一个应用数据包会被抓包后在同一个IP数据包中,因此对于使用自制dissector的时候需要考虑这种情况. Lua Dissector相关资料可以见:ht ...
- python+UIAutomation+libary
#! /usr/bin/env python#Author: XIE TIAN# -*- coding:utf8 -*-from __future__ import unicode_literalsi ...