[LUOGU] P3004 [USACO10DEC]宝箱Treasure Chest

第一眼:区间DP,可以瞎搞

f[i][j]=max(sum(i,j)-f[i+1][j],sum(i,j)-f[i][j-1])

提出来就是f[i][j]=sum(i,j)-min(f[i+1][j],f[i][j-1])

可是空间是64M,就很难受,第二个数据点是极限数据

尝试了用部分记忆化的方式找空间时间平衡,但是这样显然是不对的,看起来“省下的空间”实际上成为了更多的栈空间..

考虑写一维的DP,既然不可能消一维,就把另一维藏起来。

f[i]表示原来的f[i][i+len],外层依旧枚举len

f[i]=sum(i,j)-min(f[i],f[i+1]),等号前是要更新的len意义下的f[i][i+len],后面的f[i]和f[i+1]代表f[i][i+len-1]和f[i+1][i+len],也就是原来的f[i][j-1]和f[i+1][j]啦

是不是只依赖两层的区间DP都可以这样搞呢?类似横向的滚动数组咯?

 想了一下,是不行的。像合并果子、Polygon这种题,虽然也是小区间依赖大区间,外层枚举len,但是有一个枚举断点的操作,导致转移区间长度不具有约束关系

就好比在背包中突然要从历史版本转移,当然是不行的。这种题是转移长度确定的,才能消去一维(类似滚动)。

#include<iostream>
#include<cstdio>

using namespace std;

const int MAXN=5050;

inline int rd(){
  int ret=0,f=1;char c;
  while(c=getchar(),!isdigit(c))f=c=='-'?-1:1;
  while(isdigit(c))ret=ret*10+c-'0',c=getchar();
  return ret*f;
}

int n;
int sum[MAXN];
int f[5005];

int main(){
  n=rd();
  int x;
  for(int i=1;i<=n;i++) sum[i]=sum[i-1]+(f[i]=rd());
  for(int len=1;len<=n;len++){
    for(int i=1;i+len<=n;i++){
      int j=i+len;
      f[i]=sum[j]-sum[i-1]-min(f[i],f[i+1]);
    }
  }
  cout<<f[1];
  return 0;
}

本文来自博客园,作者:GhostCai,转载请注明原文链接:https://www.cnblogs.com/ghostcai/p/9265049.html

原文地址:https://www.cnblogs.com/ghostcai/p/9265049.html