본문 바로가기
알고리즘/백준

[Java] BOJ 9252번 LCS2

by 컴공맨 2021. 6. 25.
 

9252번: LCS 2

LCS(Longest Common Subsequence, 최장 공통 부분 수열)문제는 두 수열이 주어졌을 때, 모두의 부분 수열이 되는 수열 중 가장 긴 것을 찾는 문제이다. 예를 들어, ACAYKP와 CAPCAK의 LCS는 ACAK가 된다.

www.acmicpc.net


풀이

DP와 재귀를 사용하여 해결했습니다.


코드

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;

public class Main {
	public static String S1, S2;
	public static int[][] memo;
	public static void main(String[] args) throws IOException {
		BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
		S1 = br.readLine();
		S2 = br.readLine();
		
		int len1 = S1.length();
		int len2 = S2.length();
		
		memo = new int[len1 + 1][len2 + 1];
		
		for (int i = 1; i <= len1; ++i) {
			for (int j = 1; j <= len2; ++j) {
				
				if (S1.charAt(i - 1) == S2.charAt(j - 1)) {
					// 문자가 같다면 이전 문자까지의 LCS에 +1을 해주어야하는데
					// S1 문자열일 경우 현재 문자의 이전 문자는 i - 1이고
					// S2 문자열일 경우 현재 문자의 이전 문자는 j - 1이 되므로
					// S1, S2 둘다의 이전 문자의 LCS는 memo[i - 1][j - 1]이 된다.
					memo[i][j] = memo[i - 1][j - 1] + 1;
				}
				else {
					// 문자가 다를경우 S1 문자열의 이전 문자까지의 LCS 즉, memo[i - 1][j]와
					// S2 문자열의 이전 문자까지의 LCS 즉, memo[i][j - 1] 중 큰 값을 선택하면 된다.
					memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);
				}
			}
		}
		
		System.out.println(memo[len1][len2]);
		System.out.println(print(len1, len2));
	}
	
	public static String print(int i, int j) {
		if(i == 0 || j == 0) {
			return "";
		} else if(S1.charAt(i - 1) == S2.charAt(j - 1)) {
			return print(i - 1, j - 1) + S1.charAt(i - 1);
		} else {
			if(memo[i][j - 1] > memo[i - 1][j])
				return print(i, j - 1);
			else
				return print(i - 1, j);
		}
	}
}

 

pyo7410/Algorithm

1일 1커밋을 목표로! Contribute to pyo7410/Algorithm development by creating an account on GitHub.

github.com

 

'알고리즘 > 백준' 카테고리의 다른 글

[Java] BOJ 1068번 트리  (0) 2021.06.29
[Java] BOJ 13549번 숨바꼭질 3  (0) 2021.06.28
[Java] BOJ 9019번 DSLR  (0) 2021.06.24
[Java] BOJ 2493번 탑  (0) 2021.06.23
[Java] BOJ 16973번 직사각형 탈출  (0) 2021.06.22