티스토리 뷰
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 |
댓글