利用C++實(shí)現(xiàn)最長(zhǎng)公共子序列與最長(zhǎng)公共子串
一、問題描述
子串應(yīng)該比較好理解,至于什么是子序列,這里給出一個(gè)例子:有兩個(gè)母串
cnblogs
belong
比如序列bo, bg, lg在母串cnblogs與belong中都出現(xiàn)過并且出現(xiàn)順序與母串保持一致,我們將其稱為公共子序列。最長(zhǎng)公共子序列(Longest Common Subsequence, LCS),顧名思義,是指在所有的子序列中最長(zhǎng)的那一個(gè)。子串是要求更嚴(yán)格的一種子序列,要求在母串中連續(xù)地出現(xiàn)。在上述例子的中,最長(zhǎng)公共子序列為blog(cnblogs, belong),最長(zhǎng)公共子串為lo(cnblogs, belong)。
二、求解算法
對(duì)于母串X=<x1,x2,⋯,xm>X=<x1,x2,⋯,xm>, Y=<y1,y2,⋯,yn>Y=<y1,y2,⋯,yn> ,求LCS與最長(zhǎng)公共子串。
暴力解法
假設(shè) m<nm<n, 對(duì)于母串XX,我們可以暴力找出2m2m個(gè)子序列,然后依次在母串YY中匹配,算法的時(shí)間復(fù)雜度會(huì)達(dá)到指數(shù)級(jí)O(n∗2m)O(n∗2m) 。顯然,暴力求解不太適用于此類問題。
動(dòng)態(tài)規(guī)劃
假設(shè)Z=<z1,z2,⋯,zk>Z=<z1,z2,⋯,zk>是XX與YY的LCS, 我們觀察到
如果xm=ynxm=yn,則zk=xm=ynzk=xm=yn,有Zk−1Zk−1是Xm−1Xm−1與Yn−1Yn−1的LCS;
如果xm≠ynxm≠yn,則ZkZk是XmXm與Yn−1Yn−1的LCS,或者是Xm−1Xm−1與YnYn的LCS。
因此,求解LCS的問題則變成遞歸求解的兩個(gè)子問題。但是,上述的遞歸求解的辦法中,重復(fù)的子問題多,效率低下。改進(jìn)的辦法——用空間換時(shí)間,用數(shù)組保存中間狀態(tài),方便后面的計(jì)算。這就是動(dòng)態(tài)規(guī)劃(DP)的核心思想了。
DP 求解 LCS
用二維數(shù)組c[i][j]記錄串x1x2⋯xix1x2⋯xi與y1y2⋯yjy1y2⋯yj的LCS長(zhǎng)度,則可得到狀態(tài)轉(zhuǎn)移方程

代碼實(shí)現(xiàn)
public static int lcs(String str1, String str2) {
int len1 = str1.length();
int len2 = str2.length();
int c[][] = new int[len1+1][len2+1];
for (int i = 0; i <= len1; i++) {
for( int j = 0; j <= len2; j++) {
if(i == 0 || j == 0) {
c[i][j] = 0;
} else if (str1.charAt(i-1) == str2.charAt(j-1)) {
c[i][j] = c[i-1][j-1] + 1;
} else {
c[i][j] = max(c[i - 1][j], c[i][j - 1]);
}
}
}
return c[len1][len2];
}
DP 求解最長(zhǎng)公共子串
前面提到了子串是一種特殊的子序列,因此同樣可以用DP來解決。定義數(shù)組的存儲(chǔ)含義對(duì)于后面推導(dǎo)轉(zhuǎn)移方程顯得尤為重要,糟糕的數(shù)組定義會(huì)導(dǎo)致異常繁雜的轉(zhuǎn)移方程??紤]到子串的連續(xù)性,將二維數(shù)組c[i][j]用來記錄具有這樣特點(diǎn)的子串——結(jié)尾同時(shí)也為為串x1x2⋯xix1x2⋯xi與y1y2⋯yjy1y2⋯yj的結(jié)尾——的長(zhǎng)度。
得到轉(zhuǎn)移方程:

最長(zhǎng)公共子串的長(zhǎng)度為 max(c[i,j]), i∈{1,⋯,m},j∈{1,⋯,n}max(c[i,j]), i∈{1,⋯,m},j∈{1,⋯,n} 。
代碼實(shí)現(xiàn)
public static int lcs(String str1, String str2) {
int len1 = str1.length();
int len2 = str2.length();
int result = 0; //記錄最長(zhǎng)公共子串長(zhǎng)度
int c[][] = new int[len1+1][len2+1];
for (int i = 0; i <= len1; i++) {
for( int j = 0; j <= len2; j++) {
if(i == 0 || j == 0) {
c[i][j] = 0;
} else if (str1.charAt(i-1) == str2.charAt(j-1)) {
c[i][j] = c[i-1][j-1] + 1;
result = max(c[i][j], result);
} else {
c[i][j] = 0;
}
}
}
return result;
}
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容改了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。
相關(guān)文章
C++編寫LINUX守護(hù)進(jìn)程的實(shí)現(xiàn)代碼
這篇文章主要介紹了如何使用C++實(shí)現(xiàn)LINUX守護(hù)進(jìn)程,文中代碼非常詳細(xì),供大家學(xué)習(xí)參考,感興趣的小伙伴可以了解下2020-06-06
vc控制臺(tái)程序關(guān)閉事件時(shí)的處理方式及注意點(diǎn)詳解
在本篇文章里小編給大家整理的是一篇關(guān)于vc控制臺(tái)程序關(guān)閉事件時(shí)的正確處理方式的相關(guān)知識(shí)點(diǎn)內(nèi)容,對(duì)此有需求的朋友們可以參閱下。2021-12-12
C++?棧和隊(duì)列的實(shí)現(xiàn)超詳細(xì)解析
棧和隊(duì)列,嚴(yán)格意義上來說,也屬于線性表,因?yàn)樗鼈円捕加糜诖鎯?chǔ)邏輯關(guān)系為?"一對(duì)一"?的數(shù)據(jù),但由于它們比較特殊,因此將其單獨(dú)作為一章,做重點(diǎn)講解2022-03-03
Opencv實(shí)現(xiàn)聯(lián)合雙邊濾波
這篇文章主要為大家詳細(xì)介紹了Opencv實(shí)現(xiàn)聯(lián)合雙邊濾波,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-10-10
C語言實(shí)現(xiàn)學(xué)生學(xué)籍管理系統(tǒng)課程設(shè)計(jì)
這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)學(xué)生學(xué)籍管理系統(tǒng)課程設(shè)計(jì),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-07-07
三種獲取網(wǎng)頁源碼的方法(使用MFC/Socket實(shí)現(xiàn))
Windows下比較簡(jiǎn)單的獲取網(wǎng)頁源碼的方法:使用MFC、使用MFC、Socket實(shí)現(xiàn)2013-12-12
Qt數(shù)據(jù)庫(kù)應(yīng)用之通用數(shù)據(jù)庫(kù)同步
數(shù)據(jù)庫(kù)同步的主要功能是將本地的數(shù)據(jù)庫(kù)記錄同步到遠(yuǎn)程的數(shù)據(jù)庫(kù)。本文將利用Qt實(shí)現(xiàn)通用數(shù)據(jù)庫(kù)同步功能,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2022-03-03
C++11中跳轉(zhuǎn)initializer_list實(shí)現(xiàn)分析
這篇文章主要介紹了C++11中跳轉(zhuǎn)initializer_list實(shí)現(xiàn)分析,實(shí)例分析initializer_list<T>初體驗(yàn),結(jié)合示例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-04-04

