nefu269 noip2006能量项链【区间dp 环】

2006noip省选第一题啊,卡了我几个小时不知道错哪了了。原来还是没理解好递推的顺序。

刚刚那个合并石子   就是不理解查找某一区间最大值时后面一半没有求出的处理,这个题也一样,偏执的觉得既然寻找最大值时候只是在1-n的区间里寻找,那我的起点就只需要遍历到n就可以了,但是啊,比方说你求n~2*n-1的时候后边一半的值怎么来的啊,不还得你自己求啊

林大的网页复制下去没排版,截屏凑合看吧

nefu269 noip2006能量项链【区间dp 环】_第1张图片

还有一个结构体版的,照着网友的思路改的,其实没必要,只是清晰了一点而已

/***********
nefu269
2016.2.18
952k 	21ms	C++ (g++ 3.4.3)	1226
***********/
#include <iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int n,f[205][205];
int max(int a,int b)
{
    return a>b?a:b;
}
struct node
{
    int e,s;
}a[300];
int main()
{
  // freopen("cin.txt","r",stdin);
    while(~scanf("%d",&n))
    {
        int t;
        scanf("%d",&t);
        a[n].e=a[1].s=t;
        for(int i=2;i<=n;i++)
        {
            scanf("%d",&t);
            a[i-1].e=a[i].s=t;
        }
        for(int i=n+1;i<=2*n-1;i++)
        {
            a[i].e=a[i-n].e;
            a[i].s=a[i-n].s;
        }
        for(int i=1;i<=2*n;i++) f[i][i]=0;
        for(int i=2;i<=n;i++)//真正长度
            for(int j=1;j<=2*n-i;j++)//开头
            {
                f[j][i+j-1]=0;
                for(int k=j;k<j+i-1;k++)//中点
                f[j][i+j-1]=max(f[j][i+j-1],f[j][k]+f[k+1][i+j-1]+a[j].s*a[k].e*a[i+j-1].e);
            }

        int maxn=0;
        for(int i=1;i<=n;i++) if(maxn<f[i][i+n-1]) maxn=f[i][i+n-1];
      //  for(int i=1;i<=n;i++) for(int j=i;j<=i+n-1;j++) printf("开头:%d,结尾:%d,f=%d\n",i,j,f[i][j]);
        printf("%d\n",maxn);
    }
    return 0;
}


你可能感兴趣的:(dp)