P1063能量项链

传送

这又是一道经典的区间DP题。

复习一下区间DP的做法。

三重循环,第一层枚举区间长度,第二层枚举起点,第三层枚举断点。

区间长度是从1到n-1(因为如果是从1到n的话,1+n≠n,所以是1到n-1)。注意这里的n就是总的元素个数(一般就是题目中给出的n,而不是处理完环变链之后的总数(2*n-1))

但是在枚举起点的时候要特别注意,要使这次枚举起点的最大值+当前枚举的区间长度=链的最后一个元素的编号。

举个例子:在这道题里,我们这样枚举:

for(int len=2;len<=n;len++)//一开始我设的是end=st+len-1,不过等价于上面所说的len从1到n-1,end=st+len
        for(int st=1;st<=2*n-len+1;st++)//注意枚举的终止条件

感性理解一下ρωρ

枚举断点的时候,断点k是从st枚举到end-1(如果枚举到end,则k+1=end+1,f[end+1][end]没有实际意义)

通用的做法讲完了,来扯一下这道题

用h[i],表示i号点的头标记(head),用t[i]表示i节点的尾标记(tail),f[i][j]表示从第i个点合并到第j个点的最大得分,初始化:f[i][i]=0(因为你不需要合并,当然也没有得分)

从读入开始,就是坑。因为我们要在读入头标记的时候计算尾标记,h[i]=t[i-1]。我们注意到,点的编号从1开始,1-1是0,但是不存在0号点,所以要特判掉。并且这里还牵扯到环变链,所以也要计算h[i+n],t[i+n]。

读入完了,我们发现接下来就是一个标准的区间DP板子,只需要注意上面所提到的枚举的起止点就可以了。

递推式:f[i][j]=max(f[i][j],f[i][k]+f[k+1][j]+h[i]*t[k]*t[j])

最后答案注意不是f[1][n],而是在f[i][i+n-1](1<=i<=n)中选择最大值(不一定要从1节点开始合并)

代码:

#include<bits/stdc++.h>
using namespace std;
const int N=105;
int n,h[N*2],t[N*2],f[N*2][N*2];
int read()//快读
{
    char ch=getchar();
    int x=0;bool f=0;
    while(ch<'0'||ch>'9')
    {
        if(ch=='-')f=1;
        ch=getchar();
    }
    while(ch>='0'&&ch<='9')
    {
        x=(x<<3)+(x<<1)+(ch^48);
        ch=getchar();
    }
    f?x=-x:x=x;
    return x;
}
int main()
{
    n=read();
    for(int i=1;i<=n;i++)
    {
        h[i]=read();
        int r=i-1;if(r==0)r=n;//pay attention
        t[r]=h[i];
        h[i+n]=h[i];
        t[r+n]=t[r];
    }
    for(int len=2;len<=n;len++)//len=r-l+1
    {
        for(int st=1;st<=2*n-len+1;st++)//注意一下
        {
            int end=st+len-1;
            for(int k=st;k<end;k++)
            {
                f[st][end]=max(f[st][end],f[st][k]+f[k+1][end]+h[st]*t[k]*t[end]);
            }
        }
    }
    int maxn=-1000000;
    for(int i=1;i<=n;i++)
     maxn=max(maxn,f[i][i+n-1]);//找答案
    printf("%d",maxn); 
}

 

posted @ 2019-07-01 15:04  千载煜  阅读(130)  评论(0编辑  收藏  举报