티스토리 뷰

 

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

 

12738번: 가장 긴 증가하는 부분 수열 3

첫째 줄에 수열 A의 크기 N (1 ≤ N ≤ 1,000,000)이 주어진다. 둘째 줄에는 수열 A를 이루고 있는 Ai가 주어진다. (-1,000,000,000 ≤ Ai ≤ 1,000,000,000)

www.acmicpc.net

가장 긴 증가하는 부분 수열을 DP가 아닌 이분탐색을 활용한 방법입니다. 

 

LIS[i] = 길이 i+1 인 증가하는 부분 수열들 중 가장 큰 값 중 가장 작은 값

이라고 하면, LIS[i] < LIS[i+1] 은 자명합니다.

(가장 작은 값만 저장하게 되므로, LIS의 길이가 i+1로 증가했다는 것은 LIS[i] 보다 큰 값이 존재한다는 뜻)

 

따라서 LIS 는 오름차순 정렬이 되며, 주어진 수열의 각 값 (arr[j]) 를 LIS 에 대해 이분탐색하여

LIS에서 arr[j] 이하의 값 중 가장 작은 첫 번째 값을 찾도록 합니다.

 

그렇게 되면, LIS[i] < arr[j] 라면, LIS[i+1] = arr[j] 가 되므로 LIS의 길이가 최종적으로 가장 긴 증가하는 부분 수열의 길이가 됩니다.

 

코드는 아래와 같습니다.

#include <iostream>
#include <vector>
#include <algorithm>

using namespace std;

int n;
vector<int> vec;
vector<int> lis; 
// lis[i] : 길이 i+1인 LIS의 가장 큰 수 중 가장 작은 값
// 가장 작은 값들로 저장되므로 오름차순 정렬됨됨

void init()
{
    cin >> n;
    vec.resize(n);
    for(int i = 0; i < n; i++) {
        cin >> vec[i];
    }
    lis.resize(n);
}

void calc()
{
    int idx = 0;
    for(int i = 0; i < n; i++) {
        int pos = lower_bound(lis.begin(), lis.begin() + idx, vec[i]) - lis.begin();
        lis[pos] = vec[i];
        if(pos == idx) {
            idx++;
        }
    }
    
    cout << idx;
}

int main()
{
    ios_base :: sync_with_stdio(false);
    cin.tie(NULL);
    cout.tie(NULL);
    
    init();
    calc();
    
}

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

백준 1201 NMK C++  (0) 2023.06.27
백준 11689 GCD(n, k) = 1 C++  (0) 2023.05.02
백준 2086 피보나치 수의 합 C++  (1) 2023.04.14
백준 10220 Self Representing Seq python3  (0) 2023.03.28
백준 1939 중량제한 C++  (0) 2023.03.17
댓글
공지사항
최근에 올라온 글
최근에 달린 댓글
Total
Today
Yesterday
TAG
more
«   2025/02   »
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 26 27 28
글 보관함