uva11081 - Strings(递推)

题目:uva11081 - Strings(递推)

题目大意:给你三个字符串A,B,C。然后问你用A和B的中一些子串(可以有空串)拼出C的方法数。这里的子串指的是将A,B中去掉一些字符得到新的子串。

题目大意:这题应该和之前有一题,给出A和C,问C可以由多少的A的子串表示差不多,不一样的是这里是可以由两个串来表示了。一开始只开一个三维数组,结果发现怎么算,结果都太大了。觉得肯定是有重复计算了。后面看了题接才发现原来应该开三个三维数组。f1【i】【j】【k】代表:拼出第三个串的第K个字符时,用到了第一个串的前I个,用到了第二串的前j个,但是拼出第三个串的第K个字符用的是第一个串。f2【i】【j】【k】同理用到的是第二个串,这样f【i】【j】【k】
= f1【i】【j】】【k】 + f2【i】【j】【k】。这两个f1,f2分开来算。

f1【i】【j】【k】 =  f1【i - 1】【j】【k】;s3【k】 == s1【i】 f1【i】【j】【k】 += f【i - 1】【j】【k - 1】;(为什么加上的是f,而不是f1,因为已经确定了第k个字符是出自s1了,那么中间的那些字符是出 自哪个字符串都是可以的。这样的话不仅不会漏算,也避免了不相等情况的重复。)

f2【i】【j】【k】 = f2【i】【j - 1】【k】 ;s3【k】 == s2【j】 , f2【i】【j】【k】 += f【i】【j - 1】【k - 1】;注意递推的时候要考虑子串是空串的情况。然后就是边界:因为有空串,所以f1【i】【j】【0】 =  f2【i】【j】【0】 = f【i】【j】【0】 = 1;

代码:

#include <cstdio>
#include <cstring>

const int N = 65;
const int MOD = 10007;

char s1[N];
char s2[N];
char s3[N];

int f1[N][N][N], f2[N][N][N], f[N][N][N];
int l1, l2, l3;

void init () {

	for (int i = 0; i <= l2; i++)
		for (int j = 0; j <= l1; j++) {

			f2[0][j][i] = 1;
			f1[0][j][i] = 1;
			f[0][j][i] = 1;
		}

	for (int i = 1; i <= l3; i++)
		for (int j = 1; j <= l2; j++)
			f1[i][0][j] = 0;

	for (int i = 1; i <= l3; i++)
		for (int j = 1; j <= l1; j++)
			f2[i][j][0] = 0;
}

int main () {

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

		scanf ("%s%s%s", s1, s2, s3);
		l1 = strlen (s1);
		l2 = strlen (s2);
		l3 = strlen (s3);

		init ();

		for (int i = 1; i <= l3; i++)
			for (int j = 0; j <= l1; j++)
				for (int k = 0; k <= l2; k++) {

					if (j) {
						f1[i][j][k] = f1[i][j - 1][k];
						if (s3[i - 1] == s1[j - 1])
							f1[i][j][k] = (f1[i][j][k] + f[i - 1][j - 1][k]) % MOD;
					}
					if (k) {
						f2[i][j][k] = f2[i][j][k - 1];
						if (s3[i - 1] == s2[k - 1])
							f2[i][j][k] = (f2[i][j][k] + f[i - 1][j][k - 1]) % MOD;
					}
					f[i][j][k] = (f1[i][j][k] + f2[i][j][k]) % MOD;
//					printf ("%d %d %d %d %d\n", i, j, k, f1[i][j][k], f2[i][j][k]);
				}
		printf ("%d\n", f[l3][l1][l2]);

	}
	return 0;
}
时间: 2024-10-24 19:17:45

uva11081 - Strings(递推)的相关文章

2017中国大学生程序设计竞赛 - 女生专场 Happy Necklace(递推+矩阵快速幂)

Happy Necklace Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)Total Submission(s): 1146    Accepted Submission(s): 491 Problem Description Little Q wants to buy a necklace for his girlfriend. Necklaces are single

hdu 1207 汉诺塔II (DP+递推)

汉诺塔II Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submission(s): 4529    Accepted Submission(s): 2231 Problem Description 经典的汉诺塔问题经常作为一个递归的经典例题存在.可能有人并不知道汉诺塔问题的典故.汉诺塔来源于印度传说的一个故事,上帝创造世界时作了三根金刚石柱子,在一根柱子上从下往

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 亚洲赛区成都站的比赛,在这里,我们获得了历史性的突破,尽管只是一枚铜牌,但获奖那一刻的激动,也许将永远铭刻

hdu 2067(递推或卡特兰数【待补充】)

//解法一:递推#include<iostream> using namespace std; long long d[36][36]; int main() { for(int i=1;i<=35;i++) { d[0][i]=1; } for(int i=1;i<=35;i++) for(int j=i;j<=35;j++) { if(i==j) d[i][j]=d[i-1][j]; else d[i][j]=d[i-1][j]+d[i][j-1]; } int n; i

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

uva 1478 - Delta Wave(递推+大数+卡特兰数+组合数学)

题目链接:uva 1478 - Delta Wave 题目大意:对于每个位置来说,可以向上,水平,向下,坐标不能位负,每次上下移动最多为1, 给定n问说有多少种不同的图.结果对10100取模. 解题思路:因为最后都要落回y=0的位置,所以上升的次数和下降的次数是相同的,并且上升下降的关系满足出栈入栈的关系.即卡特兰数. 所以每次枚举i,表示有i个上升,i个下降,用组合数学枚举出位置,然后累加求和. C(2?in)?f(i)=C(2?i?2n)?f(i?1)?(n?2?i+1)?(n?2?i+2)

hdu 1249 三角形 (递推)

三角形 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 4390    Accepted Submission(s): 2957 Problem Description 用N个三角形最多可以把平面分成几个区域? Input 输入数据的第一行是一个正整数T(1<=T<=10000),表示测试数据的数量.然后是T组测试数据,每组测试数据只

递推-练习1--noi1760 菲波那契数列(2)

递推-练习1--noi1760 菲波那契数列(2) 一.心得 二.题目 1760:菲波那契数列(2) 总时间限制:  1000ms 内存限制:  65536kB 描述 菲波那契数列是指这样的数列: 数列的第一个和第二个数都为1,接下来每个数都等于前面2个数之和.给出一个正整数a,要求菲波那契数列中第a个数对1000取模的结果是多少. 输入 第1行是测试数据的组数n,后面跟着n行输入.每组测试数据占1行,包括一个正整数a(1 <= a <= 1000000). 输出 n行,每行输出对应一个输入.