HDU 1159 Common Subsequence

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1159


思路:dp 最长公共子序列问题


     Xi=x1,⋯,xi﹥即X序列的前i个字符 (1im)

        Yj=y1,⋯,yj﹥即Y序列的前j个字符 (1jn)

      假定Z=z1,⋯,zk﹥∈LCS(X , Y)

  若Xm=Yn(最后一个字符相同),则不难用反证法证明:该字符必是XY的任一最长公共子序列Z(设长

度为k)的最后一个字符,即有Zk =Xm =Yn 且显然有Zk-1LCS(Xm-1 , Yn-1)Z的前缀Zk-1Xm-1Yn-1的最长公

共子序列。

   若XmYn,则亦不难用反证法证明:要么ZLCS(Xm-1, Y),要么ZLCS(X , Yn-1)

由于ZkXmZkYn其中至少有一个必成立,若ZkYm则有ZLCS(Xm-1 , Y),类似的,

   若ZkYn 则有ZLCS(X , Yn-1)

   若Xm=Yn,则问题化归成求Xm-1Yn-1LCSLCS(X , Y)的长度等于LCS(Xm-1 , Yn-1)的长度加1

   若XmYn 则问题化归成求Xm-1YLCSXYn-1LCS

 LCS(X , Y)的长度为:max{LCS(Xm-1 , Y)的长度, LCS(X , Yn-1)的长度}LCS(Xm-1 , Y)的长度与LCS(X , Yn-1)的长度

这两个问题不是相互独立的:

两者都需要求LCS(Xm-1Yn-1)的长度,另外两个序列的LCS中包含了两个序列的前缀的LCS,故问题具有最优

子结构性质 考虑用动态规划法。

 

引进一个二维数组C,用C[i,j]记录XiYjLCS的长度如果我们是自底向上进行递推计算,那么在计算C[i,j]

之前,C[i-1,j-1], C[i-1,j]C[i,j-1]均已计算出来。此时我们

根据X[i]=Y[j]还是X[i]Y[j],就可以计算出C[i,j]

X[i]=Y[j],则执行C[i,j]C[i-1,j-1]+1;若X[i]Y[j],则根据:

C[i-1,j]C[i,j-1],则C[i,j]C[i-1,j];否则C[i,j]C[i,j-1]


代码:

#include<stdio.h>
#include<string.h>
#define N 1000
int dp[N][N];
char a[N],b[N];
int main()
{
   int la,lb,i,j;
   while(scanf("%s%s",a,b)!=EOF)
   {
     la=strlen(a);lb=strlen(b);
	 for(i=0;i<=la;i++) dp[i][0]=0;
	 for(i=0;i<=lb;i++) dp[0][i]=0;
	 for(i=0;i<la;i++)
	   for(j=0;j<lb;j++)
	   {
	     if(a[i]==b[j])
		    dp[i+1][j+1]=dp[i][j]+1;
		 else
		 {
		    if(dp[i+1][j]>dp[i][j+1])
			  dp[i+1][j+1]=dp[i+1][j];
        	   else dp[i+1][j+1]=dp[i][j+1];
		 }
	   }
	 printf("%d\n",dp[la][lb]);
   
   }
   return 0;
}



你可能感兴趣的:(HDU 1159 Common Subsequence)