uva11258- String Partition(递推)

题目:uva11258- String Partition(递推)

题目大意:给出一系列的数字,它是由很多int型的数构成的,就是中间没有加空格。所以现在问怎样拆分这些数,使得这些数之和最大。

解题思路:这里要求是int型的整数,单个数的最大的值2147483647.dp【i】【j】代表这个数字串第i个字符到第j个字符能够得到的最大的和。

dp【i】【j】 = max(dp【i】【i + k】 + dp【i + k + 1】【j】) k >= 0 && k <= j - i - 1. 因为可能最后的这个j和前面的n个字符构成整数能够得到更大的值。

例如 : 12345678910  这个数可能的划分  1| 345678912 , 12| 345678910 ,123| 45678910, 1234|5678910, 12345 | 678910, 123456|78910, 1234567|8910, 12345678|910, 123456789|10, 1234567891|0这10种划分。第一种划分是因为|后面的那个数已经超过最大值了,所以需要再划分成两个数。还有一种可能就是这个数它的长度小于10,那么这个数当然不划分最大了。如果等于10
的话,就需要看值会不会超过最大值,如果超过就需要划分。

代码:

#include <cstdio>
#include <cstring>

typedef long long ll;

const int N = 205;
const ll maxn = 2147483647;

ll dp[N][N];
char str[N];
char s1[N];
char s2[N];
int len;

ll Max (const ll a, const ll b) { return a > b ? a : b; }

ll MaxNum (int j, int l) {

	ll ans;
	memcpy (s1, str + j, sizeof (char) * (l + 1));
	s1[l + 1] = '\0';
	sscanf (s1, "%lld", &ans);
	/*if (ans > maxn) {

		memcpy (s1, str + j + 1, sizeof (str + j + 1));
		s1[j + l] += str[j];
		s1[j + l  + 1] = '\0';
		memcpy (s2, str + j, sizeof (str + j));
		s2[j + l - 1] += str[j + l];
		s2[j + l] = '\0';

		if (strcmp (s1, s2) > 0)
			sscanf (s1, "%lld", &ans);
		else
			sscanf (s2, "%lld", &ans);
	}*/
	if (ans > maxn)
		return -1;
	return ans;
}

void init () {

	for (int i = 0; i < len; i++)
		dp[i][i] = str[i] - '0';
}

int main () { 

	int t;
	ll ans;
	ll temp;
	scanf ("%d", &t);
	while (t--) {

		scanf ("%s", str);

		len = strlen (str);

		if (len < 10)
			sscanf (str, "%lld", &ans);
		else {

			init();
//			printf ("%lld\n", MaxNum (0, 9));
			for (int l = 1; l < len; l++)
				for (int i = 0; i + l < len; i++) {

					temp = -1;
					if (l < 10)
						temp = MaxNum(i, l);
					for (int k = 0; k < l; k++)
						temp = Max (temp, dp[i][i + k] + dp[i + k + 1][i + l]);
					dp[i][i + l] = temp;
				}

			ans = dp[0][len - 1];
		}

		printf ("%lld\n", ans);
	}
	return 0;
}

uva11258- String Partition(递推),布布扣,bubuko.com

时间: 2024-10-29 03:08:00

uva11258- String Partition(递推)的相关文章

Codeforces Round #526 C - The Fair Nut and String /// 组合递推

题目大意: 给定原字符序列 找出其中所有子序列满足 1.序列内字符都为a 2.若有两个以上的字符 则相邻两个字符在原序列中两者之间存在字符b 的数量 将整个字符序列用b分开 此时再得到每个b之间a的数量 即 abbgaaba 得到 v[] = { 1 0 2 1 } 此时假设到第 i-1 段 已得到在第 i-1 段内的所有方案数为 ans (长度为1.2.3.... .i-1) 则在第 i 段时 可由前一段的方案数 和 当前段数量 组合得到ans*v[ i ] (长度为2.3.4.... .i)

HDU 5863 cjj&#39;s string game ( 16年多校10 G 题、矩阵快速幂优化线性递推DP )

题目链接 题意 : 有种不同的字符,每种字符有无限个,要求用这k种字符构造两个长度为n的字符串a和b,使得a串和b串的最长公共部分长度恰为m,问方案数 分析 : 直觉是DP 不过当时看到 n 很大.但是 m 很小的时候 发现此题DP并不合适.于是想可能是某种组合数学的问题可以直接公式算 看到题解的我.恍然大悟.对于这种数据.可以考虑一下矩阵快速幂优化的DP 首先要想到线性递推的 DP 式子 最直观的想法就是 dp[i][j] = 到第 i 个位置为止.前面最长匹配长度为 j 的方案数 但是如果仔

HDU 1250 Hat&#39;s Fibonacci (递推、大数加法、string)

Hat's Fibonacci Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submission(s): 14776    Accepted Submission(s): 4923   Problem Description A Fibonacci sequence is calculated by adding the previous two members

hdu 1267 递推

下沙的沙子有几粒? Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submission(s): 4326    Accepted Submission(s): 2268 Problem Description 2005年11月份,我们学校参加了ACM/ICPC 亚洲赛区成都站的比赛,在这里,我们获得了历史性的突破,尽管只是一枚铜牌,但获奖那一刻的激动,也许将永远铭刻

NPU 2015年陕西省程序设计竞赛网络预赛(正式赛)F题 和谐的比赛(递推 ||卡特兰数(转化成01字符串))

Description 今天西工大举办了一场比赛总共有m+n人,但是有m人比较懒没带电脑,另外的n个人带了电脑.不幸的是,今天机房的电脑全坏了只能用带的电脑,一台电脑最多两人公用,确保n>=m.但是大家来的时间不同,随机次序来机房,带电脑的人直接准备比赛而没带电脑的人需要向带电脑并还没和别人公用的人求助(当然会答应).但是,如果不存在带电脑并还没和别人公用的人,那他就要等了,等是很让人头疼的,这就不和谐了,当然假如没有这样的情况发生比赛是很和谐的. Input 输入多组数据,每组数据只有一行m(

矩阵经典题目七:Warcraft III 守望者的烦恼(矩阵加速递推)

https://www.vijos.org/p/1067 很容易推出递推式f[n] = f[n-1]+f[n-2]+......+f[n-k]. 构造矩阵的方法:构造一个k*k的矩阵,其中右上角的(k-1)*(k-1)的矩阵是单位矩阵,第k行的每个数分别对应f[n-1],f[n-2],,f[n-k]的系数.然后构造一个k*1的矩阵,它的第i行代表f[i],是经过直接递推得到的.设ans[][]是第一个矩阵的n-k次幂乘上第二个矩阵,f[n]就是ans[k][1]. 注意:用__int64 #in

递推求值【快速幂矩阵】

递推求值 描述 给你一个递推公式: f(x)=a*f(x-2)+b*f(x-1)+c 并给你f(1),f(2)的值,请求出f(n)的值,由于f(n)的值可能过大,求出f(n)对1000007取模后的值. 注意:-1对3取模后等于2   输入 第一行是一个整数T,表示测试数据的组数(T<=10000)随后每行有六个整数,分别表示f(1),f(2),a,b,c,n的值.其中0<=f(1),f(2)<100,-100<=a,b,c<=100,1<=n<=10000000

Recaman&#39;s Sequence_递推

Description The Recaman's sequence is defined by a0 = 0 ; for m > 0, am = am−1 − m if the rsulting am is positive and not already in the sequence, otherwise am = am−1 + m. The first few numbers in the Recaman's Sequence is 0, 1, 3, 6, 2, 7, 13, 20, 1

ACM - 递推类题目总结

名曰递推实际上是一类DP计数问题. UVa 10081 -  Tight Words #include <iostream> #include <cstdlib> #include <cstdio> #include <algorithm> #include <cstring> #include <string> #include <vector> #include <cmath> #define mod 10