알고리즘/문제풀이

[백준 30804번] 과일 탕후루

Ohnim · 오님 2024. 6. 11. 02:27

문제 링크

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

 

문제 요약

1부터 9로 이루어진 수열이 주어진다. 이때, 연속된 부분 수열 중에서 숫자가 최대 두 개의 숫자로만 이루어진 부분 수열의 최대 길이를 구하는 문제이다.

 

문제 풀이

이 문제를 푸는 가장 효율적인 방법은 슬라이딩 윈도우 기법을 이용하는 것이다. 하지만 문제에서 주어진 제약 조건들을 잘 활용하면 조금은 더 간단한 방법으로 문제를 해결할 수 있다. 이 문제가 Division 1, Division 2에 동시에 출제된 것을 보면 다양한 접근 방법을 생각할 수 있게 의도된 문제로 보인다.

 

문제의 제약 조건을 활용하면 더욱 효과적이고 빠른 해결 방법을 찾을 수 있기 때문에, 문제를 풀기 전에 제약 조건을 먼저 확인하는 것이 중요하다는 이유를 잘 설명해 주는 모법적인(?) 문제라고 생각한다.

 

조금은 거친 방법으로 문제를 풀기

이 문제에서 주어진 두 개의 제약조건을 생각해보자. 첫 번째는 수열을 이루고 있는 숫자는 1부터 9까지라는 것이다. 두 번째는 최대 두 개의 숫자로만 이루어진 연속된 부분 수열의 최대 길이를 구해야 한다는 것이다. 이 두 가지 조건을 이용하면 우리는 조금은 거친 방법을 생각해 볼 수 있다.

 

만약에 우리가 연속된 부분 수열을 이루고 있는 숫자 두 개를 미리 정하고, 그 조건을 만족하는 연속된 부분 수열의 최대 길이를 구하는 문제를 푼다고 생각해 보자. 주어진 수열에서 숫자 \(P_1\) 또는 \(P_2\)로 이루어진 연속된 부분 수열들 중 최대 길이를 구하는 코드는 아래와 같이 작성할 수 있다.

int func(int p1, int p2) {
    for (int i = 0 ; i < n ; ++i) {
        P[i] = (A[i] == p1 || A[i] == p2);
    }
    
    int ret = 0, cnt = 0;
    
    for (int i = 0 ; i < n ; ++i) {
        if (P[i]) {
            cnt += 1;
        } else {
            cnt = 0;
        }
        
        ret = max(ret, cnt);
    }
    
    return ret;
}

그렇다면 우리가 생각해봐야 하는 \(P_1\), \(P_2\)의 쌍의 개수는 수열을 이루고 있는 숫자가 최대 9개이므로 \({}_9 \mathrm{ C }_2\)이다. 함수 func의 시간 복잡도가 \(\mathcal{O}(N)\)이기 때문에 모든 경우에 대해 func 함수를 계산해도 주어진 제한시간 내에 충분히 들어올 수 있다.

int ans = 0;
    
for (int i = 1 ; i <= 9 ; ++i) {
    for (int j = i + 1 ; j <= 9 ; ++j) {
        ans = max(ans, func(i, j));
    }
}

printf("%d\n", ans);

 

슬라이딩 윈도우 기법으로 문제 풀기

다음으로 이 문제를 효율적인 방법으로 풀 수 있는 슬라이딩 윈도우 기법을 이용한 풀이를 알아보자. 어떠한 연속된 구간 [lo, hi]가 있을 때 이 구간을 이루고 있는 숫자의 개수가 두 개보다 많다면, hi를 증가시켜도 그 구간을 이루고 있는 숫자의 개수는 두 개보다 같거나 작아질 수 없다. 하지만 lo를 증가시키면 구간의 길이가 작아지므로 그 구간을 이루고 있는 숫자의 개수는 적어도 지금보다 같거나 혹은 더 적어질 것이다. 이 아이디어를 이용하면 lo와 hi를 조정해 \(\mathcal{O}(N)\)의 시간복잡도를 이용해 문제를 해결할 수 있다.

int solve() {
    int lo = 0, ans = 0, dNums = 0;
    
    for (int hi = 0 ; hi < n ; ++hi) {
        ++F[A[hi]];
        
        if (F[A[hi]] == 1) {
            ++dNums;
        }
        
        while (dNums > 2) {
            --F[A[lo]];
            
            if (F[A[lo]] == 0) {
                --dNums;
            }
            
            ++lo;
        }
        
        ans = max(ans, hi - lo + 1);
    }
    
    return ans;
}

만약 수열을 이루고 있는 수의 종류가 많아지거나, 수열의 길이가 지금보다 더 커진다면 위에서 소개한 완전 탐색과 비슷한 방법으로는 문제를 풀기 어렵지만 슬라이딩 기법을 이용하면 해결할 수 있다.