首页 > 其他分享 >APIO 2013 T1 ROBOTS

APIO 2013 T1 ROBOTS

时间:2023-03-06 12:11:59浏览次数:65  
标签:APIO rp int ROBOTS 2013 nx dir 505 dp

每个机器人只能和相邻的机器人合并,转成人话就是任何的合并机器人只能是一段区间。而且机器人之间的行动互不干扰。

那么就是区间 \(dp\) 了。设 \(dp_{l,r,x,y}\) 表示令由 \([l,r]\) 的机器人合并成的机器人处在 \((x,y)\),推动 \([l,r]\) 内的机器人行进的最小次数。

转移的时候,我们对每个 \([l,r]\) 枚举它是由哪两块机器人合并起来的,也就是 \([l,mid]\) 和 \((mid,r]\)。然后它在哪里出现,也就是 \((x,y)\),就是 \(dp_{l,mid,x,y}\) 和 \(dp_{mid+1,r,x,y}\) 的和,这是第一步,计算机器人第一次出现的位置的答案。

首先,我们必须在预处理 \(to_{x,y,dir}\) 表示在格子 \((x,y)\) 朝着方向 \(dir\) 推动之后会到达的坐标。这个可以用记忆化搜索来完成,因为 \((x,y,dir)\) 前进一次之后会有唯一确定的新的 \((nx,ny,ndir)\),如果新的位置是障碍,就直接找到答案,否则 \(to_{x,y,dir}=to_{nx,ny,ndir}\)。复杂度是 \(O(hw)\) 的。对所有机器人通用。

然后我们考虑计算推动 \([l,r]\) 到达所有可能到达的位置的答案。第一个想法是 \(\text{Dijkstra}\),但是带 $\log $ 的话,复杂度就有点寄。那怎么办呢?看起来我们的图还挺特殊的(一个路径上的所有点都往某个点连边),难道用 \(\text{SPFA}\) 吗?

这份代码 告诉我们 \(\text{SPFA}\) 在特殊图上的效果确实不错,但是这样做就是没有发现一个特殊性质。

\(dp_{l,r,x,y}\) 的值不会超过 \(hw\)。因为一次推动的结束点最多只有这么多个!那么我们为什么还需要用堆优化 \(\text{Dijkstra}\) 呢?我们可以直接计数排序!我们对所有 \(dp_{l,r,x,y}\) 的值进行计数,然后顺着扫描。因为从当前拓展出的答案只会比更新的点更大,比被更新的原值更小,所以我们可以给每个点维护 \(del\),一旦出队则 \(del\) 为 \(1\),然后就可以进行更新,更新不需要在原位置删除,因为新的位置一定更先出队。

这样,我们就优化了这一过程的答案。不过我们的复杂度是 \(O(n^3hw)\) 的,瓶颈来自第一步枚举初始位置(但是区间 \(dp\) 的常数很小)。

这样我们就得到 \(O(n^3hw)\) 的做法,卡卡常就能通过了。

坑点:在 \(Hack\) 数据中,有可能有走不出去的环。就要在记忆搜索 \(to\) 的过程中记录当前的节点是否 \(inqueue\),如果搜索到有环,那么来的路上的所有的都是绝对不能选的。

int dx[4]={1,0,-1,0},dy[4]={0,-1,0,1};
int dp[10][10][505][505],n,w,h;
int a[505][505],cur[505][505];
int trn[505][505],inq[505][505][4];
int tox[505][505][4],toy[505][505][4],del[505][505];
vt<pii>ope[2000005];
st s;
inline void extend(){
	rep(i,0,8*w*h)ope[i].clear();
	rp(i,w)rp(j,h)del[i][j]=0;
	rp(i,w)rp(j,h)if(a[i][j]&&cur[i][j]!=1e9)ope[cur[i][j]].pb({i,j}),del[i][j]=0;
	rep(d,0,8*w*h){
		for(auto p:ope[d]){
			int x=p.first,y=p.second;
			if(del[x][y])continue;
			del[x][y]=1;
			rd(dir,4){
				int nx=tox[x][y][dir],ny=toy[x][y][dir];
				if(nx==-1)continue;
				if(cur[nx][ny]>d+1){
					cur[nx][ny]=d+1;
					ope[d+1].pb({nx,ny});
				}
			}
		}
	}
}
inline void dfs(int x,int y,int dir){
	if(tox[x][y][dir])return;
	int nx=x+dx[dir],ny=y+dy[dir],ndir=dir;
	if(!a[nx][ny]){
		tox[x][y][dir]=x;
		toy[x][y][dir]=y;
		return;
	}
	ndir=(ndir+trn[nx][ny]+4)%4;
	if(inq[nx][ny][ndir]){
		tox[x][y][dir]=-1;
		return;
	}
	inq[x][y][dir]=1;
	dfs(nx,ny,ndir);
	inq[x][y][dir]=0;
	tox[x][y][dir]=tox[nx][ny][ndir];
	toy[x][y][dir]=toy[nx][ny][ndir];
}
signed main(){
	ios::sync_with_stdio(false);
	cin.tie(0);
	cin>>n>>h>>w;
	rep(i,1,n)rep(j,1,n)rep(x,1,w)rep(y,1,h)dp[i][j][x][y]=1e9;
	rp(i,w){
		cin>>s;
		rp(j,h){
			if(s[j-1]!='x')a[i][j]=1;
			if(s[j-1]=='A')trn[i][j]=-1;
			if(s[j-1]=='C')trn[i][j]=1;
			if(isdigit(s[j-1])){
				int x=s[j-1]-'0';
				dp[x][x][i][j]=0;
			}
		}
	}
	rp(i,w)rp(j,h)if(a[i][j])rd(dir,4)if(!tox[i][j][dir])dfs(i,j,dir);
	rep(len,1,n)rep(l,1,n-len+1){
		int r=l+len-1;
		if(len>1){
			rep(mid,l,r-1){
				rp(i,w)rp(j,h)if(dp[l][mid][i][j]!=1e9&&dp[mid+1][r][i][j]!=1e9){
					dp[l][r][i][j]=min((int)dp[l][r][i][j],dp[l][mid][i][j]+dp[mid+1][r][i][j]);
				}
			}
		}
		rp(i,w)rp(j,h)cur[i][j]=dp[l][r][i][j];
		extend();
		rp(i,w)rp(j,h)dp[l][r][i][j]=cur[i][j];
	}
	int ans=1e9;
	rp(i,w)rp(j,h)ans=min(ans,(int)dp[1][n][i][j]);
	if(ans==1e9)cout<<-1<<endl;
	else cout<<ans<<endl;
	return 0;
}
//Crayan_r

标签:APIO,rp,int,ROBOTS,2013,nx,dir,505,dp
From: https://www.cnblogs.com/jucason-xu/p/17183182.html

相关文章

  • APIO 2013 T2 Toll
    想要解决这道题的最大难点是:如果我们前面插入了边的话,可能会改变树的形态,从而对后面的点的安排产生影响。每次都需要重构树形态。考虑消除顺序的影响,枚举最终加入生成树的......
  • P3558 [POI2013]BAJ-Bytecomputer
    给定一个长度为nn的只包含−1,0,1−1,0,1的数列A,每次操作可以使ai←ai+ai−1,求最少操作次数使得序列单调不降。  F[i][3] 分类讨论 #include<io......
  • 河北工程806c/c++程序设计2013年-2021年编程题
    ps:都是自己练习写的,可能不是最好的写法,但是都运行过,能跑起来。2021年1.从键盘上输入一元二次方程(ax2+bx+c=0)的系数:a,b,c;计算并输出方程的根,如果没有实根则输出“No......
  • P1983 [NOIP2013 普及组] 车站分级
    P1983[NOIP2013普及组]车站分级https://www.luogu.com.cn/problem/P1983 思路https://www.cnblogs.com/tomori/p/14331510.html   Codehttps://www.luo......
  • Gym100959I-Robots题解
    题意:平面直角坐标系上有\(n\)个机器人,每个机器人有一个上下左右之一的方向,初始所有机器人静止。在第\(0\)秒,你让第一个机器人开始朝它的方向移动,速度为每秒一个单位。......
  • Exchange 2013 清空邮箱
    在某些应用场景中,需要清空用户邮箱的所有数据。如果使用Outlookwebapp或者Outlook的邮件删除方式,对数以千计的邮件来说,实在不是一个好办法。exchange管理员可以使用“Se......
  • P3989 [SHOI2013]阶乘字符串 题解
    由于一些不可抗拒的原因,\(n\ge22\)无解。那么只用考虑\(n\le21\)的情况即可。由于\(n\)的范围缩小,导致状压又可以重新使用,所以考虑状压。设\(f_i\)为\(i\)中......
  • P3990 [SHOI2013]超级跳马
    首先可以想到一个位置状态会从与自己相差不到一行,相差列为奇数的列转移过来,那么很明显对于每一行可以求奇数位置或是偶数位置的前缀和。设\(f_{i,j}\)为跳到第\(i\)行......
  • P3997 [SHOI2013]扇形面积并 题解
    理解题意后可以把题目看成一个覆盖线段的问题。对于点在\(-m\)上,看成在\(m\)上。对于\(l<r\),不用处理。对于\(l>r\),将问题看成\((l,m)\)和\((-m+1.r)\)两个区......
  • [Sdoi2013] [bzoj 3198] spring (hash+容斥原理)
    题目描述给出个维坐标,求有多少对点对满足恰好个位置相等坐标数值在以内题目分析这道题一看就是hash容斥原理,用个位置对应相等个位置对应相等个位置对应相等的…但是不能......