BZOJ 1200 木梳

Description

 

Input

第一行为整数L,其中4≤L≤100000,且有50%的数据满足L≤104,表示木板下侧直线段的长。第二行为L个正整数A1,A2,…,AL,其中Ai≤108

Output

仅包含一个整数D,表示为使梳子面积最大,需要从木板上挖掉的格子数。

Sample Input

9
4 4 6 5 4 2 3 3 5

Sample Output

3

HINT

 

初看此题,这不是一道很水很水的dp题吗,一看数据范围马上枪毙。然后就放肆想,思考一下午未果,打了一发卡决策的dp,50分果断wa。最后还是研究题解去了。
贪心我看了很久,还是不会证明,理性的想想算了——对于某个木板的最优决策,一定存在|i-j|<=1,|b[i]-a[j]|<=2(其中b[i]指剪断后的木板高,a[i]指原木板高)。假设他是对的,那么我们dp的复杂度就会降到O(kL),其中k是一个很小的常数。
我把证明发到这里(提取码:055a),如果你看懂,我也欢迎你跟我讨论一下。
代码可能与网上其他题解的雷同,很正常,因为我是copy懂的。
 
 1 #include<cstring>
 2 #include<algorithm>
 3 #include<cstdio>
 4 #include<cstdlib>
 5 using namespace std;
 6 
 7 #define inf (1LL<<60)
 8 #define maxn 100010
 9 typedef long long ll;
10 int h[maxn],pp[maxn][32],n; ll f[maxn][2][32],ans=inf,sum;
11 
12 int main()
13 {
14     freopen("1200.in","r",stdin);
15     freopen("1200.out","w",stdout);
16     scanf("%d",&n);
17     for (int i = 1;i <= n;++i)
18     {
19         scanf("%d",h+i); sum += h[i];
20         for (int j = h[i] - 1;j <= h[i] + 1;++j)
21         {
22             pp[i][++pp[i][0]] = j;
23             if (i-1) pp[i-1][++pp[i-1][0]] = j;
24             if (i-2) pp[i-2][++pp[i-2][0]] = j;
25             if (i + 1 <= n) pp[i+1][++pp[i+1][0]] = j;
26             if (i + 2 <= n) pp[i+2][++pp[i+2][0]] = j;
27         }
28     }
29     for (int i = 1;i <= n;++i)
30     {
31         sort(pp[i]+1,pp[i]+pp[i][0]+1);
32         pp[i][0] = unique(pp[i]+1,pp[i]+pp[i][0]+1)-pp[i]-1;
33         while (pp[i][0] && pp[i][pp[i][0]] > h[i]) --pp[i][0];
34     }
35     memset(f,128,sizeof(f));
36     for (int i = 1;i <= pp[1][0];++i) f[1][0][i] = f[1][1][i] = pp[1][i];
37     for (int i = 2;i <= n;++i)
38         for (int j = 1;j <= pp[i-1][0];++j)
39             for (int k = 1;k <= pp[i][0];++k)
40             {
41                 if (pp[i-1][j]<pp[i][k])
42                     f[i][0][k] = max(f[i][0][k],f[i-1][1][j]+pp[i][k]);
43                 else if (pp[i-1][j]>pp[i][k])
44                     f[i][1][k] = max(f[i][1][k],f[i-1][0][j]+pp[i][k]);
45                 else
46                 {
47                     f[i][0][k] = max(f[i][0][k],f[i-1][0][j]+pp[i][k]);
48                     f[i][1][k] = max(f[i][1][k],f[i-1][1][j]+pp[i][k]);
49                 }
50             }
51     ans = 1LL<<60;
52     for (int p = 0;p < 2;++p)
53         for (int j = 1;j <= pp[n][0];++j)
54             ans = min(ans,sum-f[n][p][j]);
55     printf("%lld",ans);
56     fclose(stdin); fclose(stdout);
57     return 0;
58 }
View Code

 

posted @ 2015-02-10 21:40  lmxyy  阅读(321)  评论(0编辑  收藏  举报