uva10739String to Palindrome(递推)

题目:String to Palindrome

题目大意:给出一字符串,给你三种操作:可以将任何位置的字符删除,可以将任何位置的字符替换,可以在任何位置插入一个字符。问最少的操作能够把这个字符转换成回文。

解题思路:dp【i】【j】代表使字符串i到j位的子串变成回文的最少的操作。替换和删除还算好做,一开始一点都不知道插入该怎么办,后来看了别人的题解发现删除和插入是一样的效果。例如对于abbb字符串中的位置(1234)这个串,如果删除最后的那个b,那么就是dp【1】【3】 + 1.如果是在a的前面插入b,那么也是dp【1】【3】 + 1;所以这里只考虑删除。

递推的公式: s【i】 != s【j】, dp【i】【j】 = MIN (dp【i】【j - 1】 + 1, dp【i + 1】【j】 + 1,  dp【i + 1】【j - 1】+ 1)

s【i】 == s【j】, dp【i】【j】 = dp【i + 1】【j - 1】;

dp【i】【j - 1】 + 1: 把第j个字符删掉的情况。那么就是让i--j - 1的字符串成为回文的最少操作 + 1(删除操作)。

dp【i + 1】【j】 + 1:把第i个字符删掉的情况。同理。

dp【i + 1】【j - 1】  + 1 :把首位的字符其中一个替换。

计算顺序:长度小的先算。

代码:

#include <cstdio>
#include <cstring>

const int N = 1005;
int dp[N][N];
char str[N];

void init (int len) {

	for (int i = 0; i < len; i++) {

		dp[i][i] = 0;
		if (i + 1 < len) {

			if (str[i] == str[i + 1])
				dp[i][i + 1] = 0;
			else
				dp[i][i + 1] = 1;
		}
	}
}

int Min (const int a, const int b) { return a < b ? a: b; }

int main () {

	int cas;
	int len;
	scanf ("%d", &cas);
	for (int k = 1; k <= cas; k++) {

		scanf ("%s", str);
		len = strlen (str);
		init (len);

		for (int n = 3; n <= len; n++) {
			for (int i = 0, j = n - 1; j < len; i++, j++) {

				if (str[i] == str[j])
					dp[i][j] = dp[i + 1][j - 1];
				else
					dp[i][j] = Min (Min (dp[i + 1][j], dp[i][j - 1]), dp[i + 1][j - 1]) + 1; 

			}
		}

		printf ("Case %d: %d\n", k, dp[0][len - 1]);
	}
	return 0;
}

uva10739String to Palindrome(递推)

时间: 2024-07-31 15:51:22

uva10739String to Palindrome(递推)的相关文章

uva11151Longest Palindrome(递推)

题目;uva11151Longest Palindrome(递推) 题目大意:给出一个字符串,问它可以通过去掉些字符得到的最长的回文的长度. 解题思路:dp[i][j]代表从字符串i位到j位最长的回文的长度. 如果s[i] == s[j] , dp[i][j] = dp[i - 1][j - 1] + 2: 因为头尾相同,那么要求i到j的最长的回文的话,只有可能是中间的部分最长的回文加上2. 否则dp[i][j] = Max (dp[i][j - 1], dp[i + 1][j]): 计算顺序:

uva10453 - Make Palindrome(递推+ 路径输出)

题目:uva10453 - Make Palindrome(递推+ 路径输出) 题目大意:给出一字符串,仅仅只能做增加字符的操作,问最少增加多少字符串可以是的最后的字符串变成回文.并且将这样的字符串(增加长度要是最小的)的任意一种输出. 解题思路:dp[i][j]代表第i个字符到第j个字符之间要增加的最少的字符串.递推公式:s[i] == s[j], dp[i][j] = dp[ i + 1][j - 1]: s[i] != s[j] ,dp[i][j] = min (dp[i + 1][j]

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组测试数据,每组测试数据只