bzoj千题计划230:bzoj3205: [Apio2013]机器人
http://www.lydsy.com/JudgeOnline/problem.php?id=3205
历时一天,老子终于把它A了
哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈哈
因为不懂spfa的优化 以及 数组越界 TAT
┭┮﹏┭┮
牢骚发完了,题解在下面 (⊙o⊙)…
n只有9,很像状压dp
dp[l][r][x][y] 表示在(x,y)位置 合成了x-y复合机器人 的最少推动次数
它的转移 存在后效性
所以上 斯坦纳树
自身的转移:dp[l][r][x][y]=min{dp[l][k][x][y]+dp[k+1][r][x][y]}
互相转移:
预处理出 在(x,y)位置 向4个方向推,最终会停留在哪个位置
可以记忆化搜索
然后用spfa
裸的spfa 得了65
加上SLF优化得了75
再加LLL优化还是75,但慢了一点儿 (可能是我不会用吧TAT)
然后改了双端队列还是75
最后参考了一位大佬的做法:
双端队列优化只是考虑 当前要入队的和在队首的 目前哪个更优
那么可以将这个扩展成单调队列,根noip2017蚯蚓很像
两个队列q1,q2
q2中存放在自身转移中入队的状态,从小到大排好序
然后由q2中转移出的状态,全部放进q1,
因为转移代价是1,q2又单调,所以可以保证q1也是单调的
每次比较q1和q2的队首,取出目前更优的状态 扩展新的状态
#include<queue>
#include<cstdio>
#include<cstring> using namespace std; #define N 501 typedef pair<int,int> pii; #define MP(x,y) make_pair(x,y) int T,m,n;
char s[N]; int map[N][N]; int dx[]={-,,,};
int dy[]={,,,-}; int pos[][N][N];
bool v[][N][N]; int dp[][][N][N]; int tot,mx;
int cnt[N*N*];
pii tmp[N*N],q[N*N];
queue<pii>q1,q2; bool vis[N][N]; inline int &min(int &x,int &y) { return x<y ? x: y; } bool inmap(int x,int y)
{
if(x<= || x>n || y<= || y>m) return false;
if(!map[x][y]) return false;
return true;
} int find(int x,int y,int d)
{
if(pos[d][x][y]) return pos[d][x][y];
if(!map[x][y]) return -;
if(v[d][x][y]) return -;
v[d][x][y]=true;
int nd=d;
if(map[x][y]==-) nd=(d-+)%;
if(map[x][y]==-) nd=(d+)%;
int npos=-;
if(!inmap(x+dx[nd],y+dy[nd])) npos=-;
else npos=find(x+dx[nd],y+dy[nd],nd);
v[d][x][y]=false;
if(npos==-) return pos[d][x][y]=(x-)*m+y;
else return pos[d][x][y]=npos;
} void spfa(int l,int r)
{
for(int i=;i<=mx;++i) cnt[i]=;
for(int i=;i<=tot;++i) cnt[dp[l][r][tmp[i].first][tmp[i].second]]++;
for(int i=;i<=mx;++i) cnt[i]+=cnt[i-];
for(int i=tot;i;--i) q[cnt[dp[l][r][tmp[i].first][tmp[i].second]]--]=tmp[i];
for(int i=;i<=tot;++i) q2.push(q[i]);
int x,y,nx,ny;
while(!q2.empty() || !q1.empty())
{
if(q1.empty())
{
x=q2.front().first;
y=q2.front().second;
q2.pop();
}
else if(q2.empty())
{
x=q1.front().first;
y=q1.front().second;
q1.pop();
vis[x][y]=false;
}
else if(dp[l][r][q1.front().first][q1.front().second]>dp[l][r][q2.front().first][q2.front().second])
{
x=q2.front().first;
y=q2.front().second;
q2.pop();
}
else
{
x=q1.front().first;
y=q1.front().second;
q1.pop();
vis[x][y]=false;
}
for(int i=;i<;++i)
{
if(pos[i][x][y]==-) continue;
nx=(pos[i][x][y]-)/m+;
ny=pos[i][x][y]-(nx-)*m;
if(dp[l][r][x][y]+<dp[l][r][nx][ny])
{
dp[l][r][nx][ny]=dp[l][r][x][y]+;
if(!vis[nx][ny]) q1.push(MP(nx,ny)),vis[nx][ny]=true;
}
}
}
} int main()
{
scanf("%d%d%d",&T,&m,&n);
for(int i=;i<=n;++i)
{
scanf("%s",s+);
for(int j=;j<=m;++j)
if(s[j]=='A') map[i][j]=-;
else if(s[j]=='C') map[i][j]=-;
else if(s[j]>='' && s[j]<='') map[i][j]=s[j]-'';
else if(s[j]=='.') map[i][j]=;
}
for(int d=;d<;++d)
{
for(int i=;i<=n;++i)
for(int j=;j<=m;++j)
if(map[i][j]) find(i,j,d);
}
memset(dp,,sizeof(dp));
int oo=dp[][][][];
for(int i=;i<=n;++i)
for(int j=;j<=m;++j)
if(map[i][j]>= && map[i][j]<=) dp[map[i][j]][map[i][j]][i][j]=;
for(int i=T;i;--i)
for(int j=i;j<=T;++j)
{
tot=mx=;
for(int x=;x<=n;++x)
for(int y=;y<=m;++y)
{
for(int k=i;k<j;++k)
dp[i][j][x][y]=min(dp[i][j][x][y],dp[i][k][x][y]+dp[k+][j][x][y]);
if(dp[i][j][x][y]!=oo) tmp[++tot]=MP(x,y),mx=max(mx,dp[i][j][x][y]);
}
if(tot) spfa(i,j);
}
int ans=oo;
for(int i=;i<=n;++i)
for(int j=;j<=m;++j)
ans=min(ans,dp[][T][i][j]);
if(ans==oo) printf("-1");
else printf("%d",ans);
return ;
}
bzoj千题计划230:bzoj3205: [Apio2013]机器人的更多相关文章
- bzoj千题计划300:bzoj4823: [Cqoi2017]老C的方块
http://www.lydsy.com/JudgeOnline/problem.php?id=4823 讨厌的形状就是四联通图 且左右各连一个方块 那么破坏所有满足条件的四联通就好了 按上图方式染色 ...
- bzoj千题计划196:bzoj4826: [Hnoi2017]影魔
http://www.lydsy.com/JudgeOnline/problem.php?id=4826 吐槽一下bzoj这道题的排版是真丑... 我还是粘洛谷的题面吧... 提供p1的攻击力:i,j ...
- bzoj千题计划280:bzoj4592: [Shoi2015]脑洞治疗仪
http://www.lydsy.com/JudgeOnline/problem.php?id=4592 注意操作1 先挖再补,就是补的范围可以包含挖的范围 SHOI2015 的题 略水啊(逃) #i ...
- bzoj千题计划220:bzoj3938: Robot
http://www.lydsy.com/JudgeOnline/problem.php?id=3938 以时间为x轴,以距离为y轴,那么每个机器人的行走路径就是一条折线 把折线分段加入线段树里,然后 ...
- bzoj千题计划177:bzoj1858: [Scoi2010]序列操作
http://www.lydsy.com/JudgeOnline/problem.php?id=1858 2018 自己写的第1题,一遍过 ^_^ 元旦快乐 #include<cstdio> ...
- bzoj千题计划317:bzoj4650: [Noi2016]优秀的拆分(后缀数组+差分)
https://www.lydsy.com/JudgeOnline/problem.php?id=4650 如果能够预处理出 suf[i] 以i结尾的形式为AA的子串个数 pre[i] 以i开头的形式 ...
- bzoj千题计划304:bzoj3676: [Apio2014]回文串(回文自动机)
https://www.lydsy.com/JudgeOnline/problem.php?id=3676 回文自动机模板题 4年前的APIO如今竟沦为模板,,,╮(╯▽╰)╭,唉 #include& ...
- bzoj千题计划292:bzoj2244: [SDOI2011]拦截导弹
http://www.lydsy.com/JudgeOnline/problem.php?id=2244 每枚导弹成功拦截的概率 = 包含它的最长上升子序列个数/最长上升子序列总个数 pre_len ...
- bzoj千题计划278:bzoj4590: [Shoi2015]自动刷题机
http://www.lydsy.com/JudgeOnline/problem.php?id=4590 二分 这么道水题 没long long WA了两发,没判-1WA了一发,二分写错WA了一发 最 ...
随机推荐
- Gulp:插件编写入门
之前挖了个坑,准备写篇gulp插件编写入门的科普文,之后迟迟没有动笔,因为不知道该肿么讲清楚Stream这货,毕竟,gulp插件的实现不像grunt插件的实现那么直观. 好吧,于是决定单刀直入了.文中 ...
- Mvc4_语法基础介绍
@model MvcApplicationTest.Models.User @{ ViewBag.Title = "Index"; } <script type=" ...
- EntityFramework Core 2.x (ef core) 在迁移中自动生成数据库表和列说明
在项目开发中有没有用过拼音首字母做列名或者接手这样的项目? 看见xmspsqb(项目审批申请表)这种表名时是否有一种无法抑制的想肛了取名的老兄的冲动? 更坑爹的是这种数据库没有文档(或者文档老旧不堪早 ...
- Docker swarm集群搭建教程
一.什么是Swarm Swarm这个项目名称特别贴切.在Wiki的解释中,Swarm behavior是指动物的群集行为.比如我们常见的蜂群,鱼群,秋天往南飞的雁群都可以称作Swarm behavio ...
- Linux内核设计与实现 第十八章
1. 内核调试的难点 重现bug困难 调试风险比较大 定位bug的初始版本困难 2. 内核调试的工具和方法 2.1 输出 LOG 输出LOG不光是内核调试, 即使是在用户态程序的调试中, 也是经常使用 ...
- “吃神么,买神么”的第一个Sprint计划(结束)
“吃神么,买神么”项目Sprint计划 ——5.28 星期四(第八天)第一次Spring计划结束 第一阶段Spring的目标以及完成情况: 时间:5月21号~5月28号(7天) 目标:第一阶段结 ...
- PAT乙级(Basic Level)练习题-NowCoder数列总结
题目描述 NowCoder最近在研究一个数列: F(0) = 7 F(1) = 11 F(n) = F(n-1) + F(n-2) (n≥2) 他称之为NowCoder数列.请你帮忙确认一下数列中第n ...
- Beta阶段敏捷冲刺①
1.提供当天站立式会议照片一张. 每个人的工作 (有work item 的ID),并将其记录在码云项目管理中: 1.1昨天已完成的工作. 姓名 昨天已完成的工作 徐璐琳 熟悉"慧记" ...
- Linux命令(十九) 查看系统负载 uptime
一.命令介绍 Linux 系统中 uptime 命令主要用于获取主机运行时长和查询Linux系统负载等信息. uptime 命令可以显示系统已经运行了多长时间,信息显示依次为:现在时间.系统已经运行时 ...
- Python进阶-配置文件
一. 什么是配置文件?为什么要做配置文件? 将所有的代码和配置都变成模块化可配置化,这样就提高了代码的重用性,不再每次都去修改代码内部,这个就是我们逐步要做的事情,可配置化 二. 配置文件长啥样? 配 ...