区间 DP 常用模版:
所有的区间dp问题枚举时,第一维通常是枚举区间长度,并且一般 len = 1 时用来初始化,枚举从 len = 2 开始;第二维枚举起点 i (右端点 j 自动获得,j = i + len - 1)
模板代码如下:
for (int len = 1; len <= n; len++) { // 区间长度
for (int i = 1; i + len - 1 <= n; i++) { // 枚举起点
int j = i + len - 1; // 区间终点
if (len == 1) {
dp[i][j] = 初始值
continue;
}
for (int k = i; k < j; k++) { // 枚举分割点,构造状态转移方程
dp[i][j] = min(dp[i][j], dp[i][k] + dp[k + 1][j] + w[i][j]);
}
}
}
本题题解:
#include <iostream>
#include <algorithm>
using namespace std;
const int N = 310;
int n;
int s[N];
int f[N][N];
int main()
{
scanf("%d", &n);
for (int i = 1; i <= n; i ++ ) scanf("%d", &s[i]);//读入
for (int i = 1; i <= n; i ++ ) s[i] += s[i - 1];//前缀和
//不从1开始是因为如果len=1。那么就一堆不用合并体力消耗为0
for (int len = 2; len <= n; len ++ )//枚举长度
for (int i = 1; i + len - 1 <= n; i ++ )//枚举左端点
{
int l = i, r = i + len - 1;//右端点自动出来
f[l][r] = 1e8;//设个大值,取min
for (int k = l; k < r; k ++ ) //s[r]-s[l-1] 是从l到r这一段的和,也就是合并这两段消耗的体积
f[l][r] = min(f[l][r], f[l][k] + f[k + 1][r] + s[r] - s[l - 1]);
}
printf("%d\n", f[1][n]);//从1~n的区间
return 0;
}