其实这题前前后后的思考时间加起来应该有两天之久了,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机器人的更多相关文章

  1. bzoj3205 [Apio2013]机器人

    3205: [Apio2013]机器人 Time Limit: 15 Sec  Memory Limit: 128 MBSubmit: 953  Solved: 227[Submit][Status] ...

  2. [Bzoj3205][Apio2013]机器人(斯坦纳树)(bfs)

    3205: [Apio2013]机器人 Time Limit: 15 Sec  Memory Limit: 128 MBSubmit: 977  Solved: 230[Submit][Status] ...

  3. [APIO2013]机器人(斯坦纳树)

    题目描述 VRI(Voltron 机器人学会)的工程师建造了 n 个机器人.任意两个兼容的机 器人站在同一个格子时可以合并为一个复合机器人. 我们把机器人用 1 至 n 编号(n ≤ 9).如果两个机 ...

  4. bzoj千题计划230:bzoj3205: [Apio2013]机器人

    http://www.lydsy.com/JudgeOnline/problem.php?id=3205 历时一天,老子终于把它A了 哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈 因为不懂spfa ...

  5. [APIO2013]机器人[搜索、斯坦纳树]

    题意 题目链接 分析 记 g(d,x,y) 表示从 (x,y) 出发,方向为 d 到达的点,这个可以通过记忆化搜索求出,注意如果转移成环(此时向这个方向走没有意义)要特判. 记 f(l,r,x,y) ...

  6. BZOJ3205/UOJ107 [Apio2013]机器人

    本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...

  7. [APIO2013]机器人

    题目描述 VRI(Voltron 机器人学会)的工程师建造了 n 个机器人.任意两个兼容的机 器人站在同一个格子时可以合并为一个复合机器人. 我们把机器人用 1 至 n 编号(n ≤ 9).如果两个机 ...

  8. bzoj 3205: [Apio2013]机器人【dfs+斯坦纳树+spfa】

    第一次听说斯坦纳树这种东西 先dfs预处理出来dis[i][j][k]表示格子(i,j)向k方向转移能到哪,记忆话搜索预处理,注意如果有环的话特判一下 设f[i][j][x][y]表示复合机器人i-j ...

  9. 【BZOJ3205_洛谷3638】[APIO2013]机器人(动态规划)

    题目: 洛谷3638 分析: 卡了一天的神题--(OrzJumpmelon) 首先预处理出从点\(p\)向\(d\)方向出发最终能到达的点\(nxt[p][d]\).这个可以直接记忆化搜索解决.如果出 ...

随机推荐

  1. JDK8 新特性

    JDK8 新特性目录导航: Lambda 表达式 函数式接口 方法引用.构造器引用和数组引用 接口支持默认方法和静态方法 Stream API 增强类型推断 新的日期时间 API Optional 类 ...

  2. UVA - 12230

    #include <bits/stdc++.h> using namespace std; int n; double d; double p,l,v,ret,sum; ; /* 村庄A, ...

  3. python2.7练习小例子(九)

        9)1.题目:暂停一秒输出.     程序分析:使用 time 模块的 sleep() 函数.     程序源代码: #!/usr/bin/python # -*- coding: UTF-8 ...

  4. Xshell启动时显示丢失MSVCP110.dll解决方法

    成功安装xshell之后,在运行时却弹出“无法启动此程序,因为计算机中丢失MSVCP110.dll.尝试重新安装该程序以解决此问题”,很多人按照提示重装了还是出现同样的问题,本集教程将具体讲解如何处理 ...

  5. 【转】Python 数据库连接池

    python编程中可以使用pymysql进行数据库连接及增删改查操作,但每次连接mysql请求时,都是独立的去请求访问,比较浪费资源,而且访问数量达到一定数量时,对mysql的性能会产生较大的影响.因 ...

  6. 一步一步学Linq to sql(二):DataContext与实体

    DataContext DataContext类型(数据上下文)是System.Data.Linq命名空间下的重要类型,用于把查询句法翻译成SQL语句,以及把数据从数据库返回给调用方和把实体的修改写入 ...

  7. 搭建cvs服务器

    http://zhangjunhd.blog.51cto.com/113473/78595 http://www.cnblogs.com/lee/archive/2008/10/22/1317226. ...

  8. ActiveMQ测试实例

    ActiveMQ的安装与启动 1 下载ActiveMQ:http://activemq.apache.org/download.html 2 下载后解压到任意文件夹,解压后文件夹内的目录为: 3 进入 ...

  9. Java byte 位移操作 注意事项

    转自:http://blog.163.com/pilgrim_yang/blog/static/55631481201111542151582/ Java对byte 的 + - * / >> ...

  10. 《剑指offer》题解

    有段时间准备找工作,囫囵吞枣地做了<剑指offer>提供的编程习题,下面是题解收集. 当初没写目录真是个坏习惯(-_-)||,自己写的东西都要到处找. 提交的源码可以在此repo中找到:h ...