区间DP

1) 石子归并问题

描述: 有N堆石子排成一排,每堆石子有一定的数量。现要将N堆石子并成为一堆。合并的过程只能每次将相邻的两堆石子堆成一堆,每次合并花费的代价为这两堆石子的和,经过N-1次合并后成为一堆。求出总的代价最小值。


分析:要求n个石子归并,我们根据dp的思想划分成子问题,先求出每两个合并的最小代价,然后每三个的最小代价,依次知道n个。

定义状态dp [ i ] [ j ]为从第i个石子到第j个石子的合并最小代价。

那么dp [ i ] [ j ] = min(dp [ i ] [ k ] + dp [ k+1 ] [ j ]) 

那么我们就可以从小到大依次枚举让石子合并,直到所有的石子都合并。

#include<iostream>
#include<cstring>
using namespace std;
int a[101],dp[101][101],sum[101];
int main()
{
	int n;
	cin>>n;
	for(int i=0;i<n;i++)
	{
		cin>>a[i];
		sum[i]=(i>0?sum[i-1]:0)+a[i];
	}
	int j;
	for(int l=1;l<n;l++)
	{
		for(int i=0;i<n-l;i++)
		{
			j=i+l;
			dp[i][j]=65535;
			for(int k=i;k<j;k++)
			{
				if(dp[i][j]>dp[i][k]+dp[k+1][j]+sum[j]-(i>0?sum[i-1]:0))
					dp[i][j]=dp[i][k]+dp[k+1][j]+sum[j]-(i>0?sum[i-1]:0);
			}
		}
	}
	cout<<dp[0][n-1]<<endl;
	return 0;
}

2)整数划分问题

题目描述:给出两个整数 n , m ,要求在 n 中加入m - 1 个乘号,将n分成m段,求出这m段的最大乘积


分析:根据区间dp的思想,我们定义dp [ i ] [ j ]为从开始到 i 中加入 j 个乘号得到的最大值。

那么我们可以依次计算加入1----m-1个乘号的结果

而每次放入x个乘号的最大值只需枚举第x个乘号的放的位置即可

dp [ i ] [ j ]  = MAX (dp [ i ] [ j ] , dp [ k ] [ j-1 ] * a [ k+1 ] [ i ] ) ;

  1. #include <cstdio>  
  2. #include <cstring>  
  3. #define MAX(a,b) a>b?a:b  
  4. long long a[20][20];  
  5. long long dp[25][25];  
  6. int main()  
  7. {  
  8.     int T,m;  
  9.     scanf("%d",&T);  
  10.     getchar();  
  11.     while(T--)  
  12.     {  
  13.         char s[22];  
  14.         scanf("%s",s+1);  
  15.         scanf("%d",&m);  
  16.         int l=strlen(s),ok=1;  
  17.         memset(a,0,sizeof(a));  
  18.         for(int i=1;i<l;i++)  
  19.         {  
  20.             if(s[i]=='0')  
  21.                 ok=0;  
  22.             for(int j=i;j<l;j++)  
  23.             {  
  24.                 a[i][j]=a[i][j-1]*10+(s[j]-'0');  
  25.             }  
  26.         }  
  27.         if(ok==0&&l-1==m||l-1<m)  
  28.         {  
  29.             printf("0\n");continue;  
  30.         }  
  31.         long long x,ans;  
  32.         memset(dp,0,sizeof(dp));  
  33.         for(int i=0;i<l;i++)  
  34.             dp[i][1]=a[1][i];  
  35.         ans=0;  
  36.         if(m==1)  
  37.             ans=dp[l-1][1];  
  38.         for(int j=2;j<=m;j++)  
  39.         {  
  40.             for(int i=j;i<l;i++)  
  41.             {  
  42.                 ans=a[i][i];  
  43.                 for(int k=1;k<i;k++)  
  44.                 {  
  45.                     dp[i][j]=MAX(dp[i][j],dp[k][j-1]*a[k+1][i]);  
  46.                 }  
  47.             }  
  48.         }  
  49.         printf("%lld\n",dp[l-1][m]);  
  50.     }  
  51.     return 0;  
  52. }  
这个是借鉴的,自己确实不会。



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
区间DP是一种动态规划的方法,用于解决区间范围内的问题。在Codeforces竞赛中,区间DP经常被用于解决一些复杂的字符串或序列相关的问题。 在区间DP中,dp[i][j]表示第一个序列前i个元素和第二个序列前j个元素的最优解。具体的转移方程会根据具体的问题而变化,但是通常会涉及到比较两个序列的元素是否相等,然后根据不同的情况进行状态转移。 对于区间长度为1的情况,可以先进行初始化,然后再通过枚举区间长度和区间左端点,计算出dp[i][j]的值。 以下是一个示例代码,展示了如何使用区间DP来解决一个字符串匹配的问题: #include <cstdio> #include <cstring> #include <string> #include <iostream> #include <algorithm> using namespace std; const int maxn=510; const int inf=0x3f3f3f3f; int n,dp[maxn][maxn]; char s[maxn]; int main() { scanf("%d", &n); scanf("%s", s + 1); for(int i = 1; i <= n; i++) dp[i][i] = 1; for(int i = 1; i <= n; i++) { if(s[i] == s[i - 1]) dp[i][i - 1] = 1; else dp[i][i - 1] = 2; } for(int len = 3; len <= n; len++) { int r; for(int l = 1; l + len - 1 <= n; l++) { r = l + len - 1; dp[l][r] = inf; if(s[l] == s[r]) dp[l][r] = min(dp[l + 1][r], dp[l][r - 1]); else { for(int k = l; k <= r; k++) { dp[l][r] = min(dp[l][r], dp[l][k] + dp[k + 1][r]); } } } } printf("%d\n", dp[n]); return 0; } 希望这个例子能帮助你理解区间DP的基本思想和应用方法。如果你还有其他问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

没想好叫什么名字

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值