我是靠谱客的博主 直率小猫咪,最近开发中收集的这篇文章主要介绍区间DP学习篇(整数拆分 + 最优三角剖分),觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

题目链接

整数划分(四)

时间限制:1000 ms  |  内存限制:65535 KB

难度:3

输入

第一行是一个整数T,表示有T组测试数据
接下来T行,每行有两个正整数 n,m ( 1<= n < 10^19, 0 < m <= n的位数);

输出

输出每组测试样例结果为一个整数占一行

样例输入

2
111 2
1111 2

样例输出

11
121

 

这道题给出的思路是这样 : 理用dp[i][j]表示从1-i区间内用j个乘号的最大值,巧妙地将问题中两个条件同时简化(区间长度,乘号数量)从而实现子问题建立,需要处理num[i][j]数组保存区间i-j的数便于遍历时利用。难就难在动归思想还没掌握熟练,感觉还是积累得太少,状态转移方程 : dp[i][j] = max(dp[i][j], dp[k][j-1]*num[k+1][i])

#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
#include<vector>
#include<set>
#include<map>
#include<queue>
#include<cmath>
using namespace std;
long long dp[25][25];
char a[25];
long long num[25][25];
int main()
{
int m;
int t;
scanf("%d",&t);
while(t --)
{
scanf("%s%d",a,&m);
memset(dp, 0, sizeof(dp));
int n = strlen(a);
for(int i = 0; i < n; i ++)
{
num[i+1][i+1] = (a[i] - '0');
for(int j = i + 1; j < n; j ++)
num[i+1][j+1] = num[i+1][j] * 10 + (a[j] - '0');
}
for(int i = 1; i <= n; i ++)
dp[i][0] = num[1][i];
for(int j = 1; j < m; j ++)
for(int i = j + 1; i <= n; i ++)
for(int k = j; k < i; k ++)
dp[i][j] = max(dp[i][j], dp[k][j-1]*num[k+1][i]);
printf("%lldn",dp[n][m-1]);
}
return 0;
}

凸多边形三角划分问题

 

Problem Description

给定一个具有N(N<=50)个顶点(从1到N编号)的凸多边形,每个顶点的权值已知。问如何把这个凸多边形划分成N-2个互不相交的三角形,使得这些三角形顶点的权值的乘积之和最小。

Input

第一行为顶点数N,第二行为N个顶点(从1到N)的权值。

Output

乘积之和的最小值。题目保证结果在int范围内。

Sample Input

5

1 6 4 2 1

5

121 122 123 245 231

Sample Output

34

12214884

【思路】

用dp[i][j]表示从顶点i到顶点j的凸多边形三角剖分后所得到的最大乘积。

那么可以写出状态转移方程,并通过枚举分割点来转移。

dp[i][j]=min(dp[i][j],dp[i][k]+dp[k][j]+a[i]*a[k]*a[j])

很神奇地把问题转化成了若干个子问题,且符合动态规划三个条件

解决所有动归问题的关键,就在于寻找最优子方案,然后很容易地能写出状态转移方程啊!!!

如何寻求最优子方案?—> 个人感觉就是把所有限制条件(如序列长度,乘号数量之类的给定变量)通过dp数组进行约束,从而实现子问题的转化,转化过程中需要注意无后效性,最优策略,讲到底如何把大问题拆分成小问题,是一个思维习惯问题,多想想!!!

代码就不贴了,找不到题,思路有了一切很easy。。。

 

最后

以上就是直率小猫咪为你收集整理的区间DP学习篇(整数拆分 + 最优三角剖分)的全部内容,希望文章能够帮你解决区间DP学习篇(整数拆分 + 最优三角剖分)所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(39)

评论列表共有 0 条评论

立即
投稿
返回
顶部