최장 공통 부분 수열 알고리즘 [Longest Common Subsequence]
최장 공통 부분 수열 알고리즘 [Longest Common Subsequence]
LCS (Longest Common Subsequence)
ACAYKP, CAPCAK이 있을 때 최장 공통 부분 수열을 구하려면 다음과 같다.
A | C | A | Y | K | P | |
---|---|---|---|---|---|---|
C | 0 | 1 | 1 | 1 | 1 | 1 |
A | 1 | 1 | 2 | 2 | 2 | 2 |
P | 1 | 1 | 2 | 2 | 2 | 3 |
C | 1 | 2 | 2 | 2 | 2 | 3 |
A | 1 | 2 | 3 | 3 | 3 | 3 |
K | 1 | 2 | 3 | 3 | 4 | 4 |
arr1 = ACAYKP, arr2 = CAPCAK 일 때
i, j >= 1일 때, arr1[i-1] == arr2[j-1] 일 때 dp[i][j] = dp[i-1][j-1] + 1;
- 각 문자열의 i, j번째까지 보았을 때 같다면, 그 전까지의 문자열에서 나온 공통 부분수열 길이에 +1 해주는 것임.
arr1[i-1] != arr2[j-1] 일 때 dp[i][j] = max(dp[i][j-1], dp[i-1][j]);
- 다르다면, 여태까지 나온 부분수열의 길이 중 최고 길이를 넣어주는 것임.
- Link
- 백준 9251번 : LCS
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
#include <iostream>
#include <string>
using namespace std;
typedef long int li;
li dp[1001][1001];
int main() {
string arr1, arr2;
cin >> arr1 >> arr2;
li i=0, j=0;
for(; i<arr1.size(); i++) {
for(j=0; j < arr2.size(); j++) {
if(arr1[i] == arr2[j]) {
dp[i+1][j+1] = dp[i][j]+1;
}
else {
dp[i+1][j+1] = max(dp[i][j+1], dp[i+1][j]);
}
}
}
cout << dp[i][j] << '\n';
return 0;
}
정리글
위 링크에서 최장 공통 부분 수열 찾는 방법에 대해서도 알려줌.
This post is licensed under CC BY 4.0 by the author.