2006noip省选第一题啊,卡了我几个小时不知道错哪了了。原来还是没理解好递推的顺序。
刚刚那个合并石子 就是不理解查找某一区间最大值时后面一半没有求出的处理,这个题也一样,偏执的觉得既然寻找最大值时候只是在1-n的区间里寻找,那我的起点就只需要遍历到n就可以了,但是啊,比方说你求n~2*n-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;
}