코테/백준

99클럽 코테 스터디 32일차 TIL DP(가장 긴 바이토닉 부분 수열)

zsunny 2024. 11. 29. 01:21

https://www.acmicpc.net/problem/11054

 

<아이디어>

결국은 가장 긴 오름차순 수열과 가장 긴 내림차순 수열.

근데 이제 오름차순의 끝나는 수와 내림차순의 시작하는 수가 동일할 때의 최대 길이를 구하는 문제이다.

1. 가장 긴 증가하는 부분 수열을 찾아 dpUp배열에 저장한다.

2. 가장 긴 감소하는 부분 수열을 찾아 dpDown배열에 저장한다. (이때 배열의 끝부터 역으로 오름차순 탐색했다.)

3. 각 위치에서 dpUp + dpDown - 1 (현재 위치 수 중복) 을 계산하며 최댓값을 갱신한다.

 

<제출코드>

package Study.Week05;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.Arrays;
import java.util.StringTokenizer;

public class Day32_BOJ_11054_가장긴바이토닉부분수열 {
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        int n = Integer.parseInt(br.readLine());

        int[] arr = new int[n];
        StringTokenizer st = new StringTokenizer(br.readLine());
        for(int i=0; i<n; i++){
            arr[i] = Integer.parseInt(st.nextToken());
        }

        // 가장 긴 증가하는 부분 수열
        int[] dpUp = new int[n];
        Arrays.fill(dpUp, 1);
        for(int i=0; i<n; i++){
            for(int j=0; j<i; j++){
                if(arr[j] < arr[i]) dpUp[i] = Math.max(dpUp[i], dpUp[j]+1);
            }
        }

        // 가장 긴 감소하는 부분 수열 찾기
        int[] dpDown = new int[n];
        Arrays.fill(dpDown, 1);
        for(int i=n-1; i>=0; i--){
            for(int j=n-1; j>i; j--){
                if(arr[j] < arr[i]) dpDown[i] = Math.max(dpDown[i], dpDown[j]+1);
            }
        }

        // 가장 긴 길이 찾기
        int ans = 0;
        for(int i=0; i<n; i++){
            ans = Math.max(ans, dpUp[i] + dpDown[i] - 1);
        }
        System.out.println(ans);
    }
}