zoukankan      html  css  js  c++  java
  • BZOJ4409 [Usaco2016 Feb]Circular barn 动态规划 斜率优化

    原文链接http://www.cnblogs.com/zhouzhendong/p/8724739.html

    题目传送门 - BZOJ4409

    题意

      有一个N个点的环,相邻两个点距离是1。点顺时针标号为1..N。最初每一个点是空的。要求最终点i存在ri头牛。你有∑ri头牛。你可以选择最多k个点,然后把你的牛任意分配在这k个点里。之后,每一头牛可以选择不动,也可以顺时针走d格并呆在那里。这样,它要耗费d的能量。通过合理选择点、合理分配牛、合理安排牛的走动,使得消耗的总能量最小。

      $nleq 1000,kleq 7,r_ileq 10^6$

    题解

      首先,我们来说一个比较simple的结论。

      原始分配方案的一个点的牛不可能走到下一个点。

      很显然,如果可以走到下一个点,那么直接分配在下一个点更优。

      于是我们发现这就是分段贡献。

      考虑先断环为链,所以我们先用掉一层循环,来枚举环的开头。(事实上我是通过顺时针旋转数列实现的)

      然后考虑到剩下的部分,是个DP。

      很容易写出方程:(为了方便,这里的$a_i$即题目描述的$r_i$)

      $$dp_{r,i}=min{dp_{r-1,j}+sum_{k=j+1}^i (k-j-1)a_k} (0leq j<i)$$

      然后就是经典的斜率优化套路了。

      关于DP的斜率优化看这里$longrightarrow$传送门

      可以参照下面两道题的做法,这里我不再赘述了。

      BZOJ1096

      BZOJ3675

      然后注意一下初始的时候的$dp_{0,i}$的值为$infty$(BZOJ3675里面求的是最大值,故初始化为0;但这里求的是最小值。)。

      时间复杂度$O(n^2k)$。

    代码

    #include <bits/stdc++.h>
    using namespace std;
    typedef long long LL;
    const int N=1005;
    int n,R,q[N],head,tail;
    LL a[N],sum[N],vsum[N],x[N],y[N],dp[10][N],ans=1LL<<60;
    int main(){
    	scanf("%d%d",&n,&R);
    	for (int i=1;i<=n;i++)
    		scanf("%lld",&a[i]);
    	for (int _i_=1;_i_<=n;_i_++){
    		sum[0]=vsum[0]=0;
    		for (int i=1;i<=n;i++)
    			sum[i]=sum[i-1]+a[i],vsum[i]=vsum[i-1]+a[i]*(i-1);
    		for (int i=1;i<=n;i++)
    			dp[0][i]=1LL<<45;
    		dp[0][0]=0;
    		for (int r=1;r<=R;r++){
    			for (int i=0;i<=n;i++)
    				x[i]=i,y[i]=dp[r-1][i]-vsum[i]+sum[i]*i;
    			head=1,tail=0;
    			q[++tail]=0;
    			for (int i=1;i<=n;i++){
    				int j=q[head+1],k=q[head];
    				while (tail-head>0&&y[j]-y[k]<=sum[i]*(x[j]-x[k]))
    					head++,j=q[head+1],k=q[head];
    				j=k;
    				dp[r][i]=dp[r-1][j]+vsum[i]-vsum[j]-(sum[i]-sum[j])*j;
    				j=q[tail],k=q[tail-1];
    				while (tail-head>0&&(y[i]-y[j])*(x[j]-x[k])<=(y[j]-y[k])*(x[i]-x[j]))
    					tail--,j=q[tail],k=q[tail-1];
    				q[++tail]=i;
    			}
    			ans=min(ans,dp[r][n]);
    		}
    		for (int i=1;i<=n;i++)
    			a[i-1]=a[i];
    		a[n]=a[0];
    	}
    	printf("%lld",ans);
    	return 0;
    }
    

      

  • 相关阅读:
    111
    python 错误宝典
    Node.js Web开发:Connect
    Node.js 中的重要API:HTTP
    Node.js 中的重要API:TCP
    Node.js 中的重要API:命令行工具以及FS API 首个Node应用
    Node.js 中的JS
    Node.js 阻塞式IO与非阻塞式IO与错误处理
    Learning Vue.js 2
    A1046——入门模拟 Shortest Distance
  • 原文地址:https://www.cnblogs.com/zhouzhendong/p/BZOJ4409.html
Copyright © 2011-2022 走看看