能量项链:
题目大意:有n个珠子串成一个环,每个珠子有头标记和尾标记,每次可以合并任意相邻的两颗珠子i,j,所得的能量为head[i]*head[j]*tail[j],求将n颗珠子合并成一颗的最大能量和。n<=100.
题解:
对于有环的题目,我们一般先拆环为链,变成一条长度为2n的链。然后我们发现最终答案就是在这条长度为2n的链上找到一段长度为n的区间进行操作使得能量和最大。不难想到对于长度为n的区间我们可以看作是找到一段长度为n-1的区间进行操作使得能量和最大,再将这个能量和加上最后两颗珠子合并所得到的能量。同理可以推出n-1,n-2,n-3….2。典型的具有无后效性,于是我们采取DP解题。
由于我们刚才推出的结论与长度有关,因此我们将长度作为第一维循环,于是当我们循环到i时,我们必定知道1~i-1时的每个最优值,于是就转化为经典的区间DP了。
设f[j][j+i-1]代表从第j位到第j+i-1位的最大能量和,则
f[j][j+i-1]=max(f[j][k]+f[k+1][j+i-1]+a[j]*a[k+1]+a[j+i])(j<=k)
#include<cstdio>
#include<algorithm>
using namespace std;
int n,i,j,k,f[210][210],a[210],ans;
int main(){
scanf("%d",&n);
for(i=1;i<=n;i++)scanf("%d",&a[i]);
for(i=n+1;i<=n*2;i++)a[i]=a[i-n];
for(i=2;i<=n;i++)
for(j=1;j<=2*n-i;j++)
for(k=j;k<j+i-1;k++)
f[j][j+i-1]=max(f[j][j+i-1],f[j][k]+f[k+1][j+i-1]+a[j]*a[k+1]*a[j+i]);
for(i=1;i<=n;i++)ans=max(ans,f[i][i+n-1]);
printf("%d",ans);
return 0;
}
金明的预算方案:
题目大意:有n个物品,每个物品有一个价格和重要度,并且有可能有附件(即买了该物品才能买附件),求用不超过m的钱使得所有购买物品的价格*重要度之和最大,n<=60,m<=32000,所有价格(包括m)均为10的倍数,每个物品最多为两个物品的必需品。
题解:
显然可以看出这是一道裸的背包,因为有附件的存在,我们可以用树依赖背包或者普通的带依赖品的背包做,由于这里每个物品最多为两个物品的必需品,我们可以选择用普通的依赖背包做。
由于所有价格都是10的倍数,我们在DP的时候先把所有价格都除以10,最后输出的时候将答案乘10即可。
其