《动态规划》之--字符串比较问题(扩展距离)

问题:

对于长度相同的2个字符串AB,其距离定义为相应位置字符距离之和。2个非空格字符的距离是它们的ASCII码之差的绝对值;空格与空格的距离为0,空格与其他字符的距离为一个定值k。在一般情况下,字符串AB的长度不一定相同。字符串A的扩展是在A中插入若干空格字符所产生的字符串。在字符串AB的所有长度相同的扩展中,有一对距离最短的扩展,该距离称为字符串AB的扩展距离。对于给定的字符串AB,设计一个算法,计算其扩展距离。

测试数据:

输入:cmc      snmn        2       (分别表示字符串AB和定值k

输出:10

 

解答:

设字符串AB的字串A[1...i]B[1...j]的扩展距离是val(i, j)

依题意,字符串AB有三种可能的情况:

1A串最后一个字符是空格,B串最后一个字符是字母,则val(i, j) = val(i-1, j) + k

2A串最后一个字符时字母,B串最后一个字符时空格,则val(i, j) = val(i, j-1) + k

3A串和B串最后一个字符均是字母,则val(i, j) = val(i-1, j-1) + dist(ai , bi)

由上可知,val(i, j)具有最优子结构性质,且满足如下递推式:

val(i, j) = min{ val(i-1, j) + kval(i, j) + kval(i-1, j-1) + dist(ai , bi) }

使用动态规划算法,自底向上的计算各个子问题并利用每次计算的结果,避免重复运算,从而降低算法复杂度。

从动态规划递归式可知,算法的时间复杂度为Omn),mn分别是字符串AB的长度。

 

代码如下:

#include <iostream>

#include <cmath>

 

#define MAX 100000    //标识最大的可能整数

 

int val[300][300];

 

std::string stra;        //字符串A

std::string strb;       //字符串B

int k;        //定值k

 

//返回字符abASCII码的差的绝对值

int dist(char a, char b)

{

         return abs(a-b);     

}

 

int comp()

{

         int len1, len2;

         int tmp;

         val[0][0] = 0;

         len1 =    stra.length();

         len2 = strb.length();

        

         for(int i=0; i<=len1; i++)   //字符串AB的有效下标是º1~len,下标0表示空字符串

         {                                                                //ij0表示AB串为空串

                   for(int j=0; j<=len2; j++)

                   {

                            if(i+j)//ij至少一个大于0

                            {

                                     val[i][j] = MAX;

                                    

                                     tmp = val[i-1][j] + k;

                                     if(i && (tmp<val[i][j]))//i大于0

                                               val[i][j] = tmp;

                                              

                                     tmp = val[i][j-1]+k;

                                     if(j && (tmp<val[i][j]))//j大于0

                                               val[i][j] = tmp;

                                              

                                     tmp = val[i-1][j-1] + dist(stra[i], strb[j]);

                                     if((i*j) && (tmp<val[i][j])) //ij至少有一个不为0

                                               val[i][j] = tmp;                                          

                            }

                   }       

         }

         return val[len1][len2];

}

 

int main()

{

         std::cin>>stra>>strb>>k;

         stra = " " + stra;      //此处在字符串开头添加一个空格,是为了使字符串stra

         strb = " " + strb;    //的控制台输入的有效字符下标从1stra.length()

         std::cout<<comp()<<std::endl;

         system("pause");

         return 0;

}

 

 

 

  • 4
    点赞
  • 34
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值