백준 17076번 망가진 데이터
문제
택희는 이번 대회에 사용하기 위해 그래프를 몇 개 만들었다.
택희의 그래프 데이터는 항상 아래의 형식을 따른다.
N M U1 V1 U2 V2 U3 V3 ... UM VM
위는 정점이 N개, 간선이 M개인 그래프 데이터로, 모든 Ui, Vi에 대해 1 ≤ Ui, Vi ≤ N을 만족하며, N과 M 뒤에 등장하는 간선 정보(U, V 페어)는 정확히 M개이다. 그 외의 다른 조건은 없다.
택희는 대회에 사용할 모든 그래프를 만들었고, 데이터를 업로드하려던 그 순간! 갑자기 영훈이가 나타나 데이터의 여기저기에 임의의 정수를 몇 개 써 넣어 버렸고, 택희는 순간적인 상황 변화를 받아들이지 못하고 격한 욕설을 내뱉으며 영훈이에게 당장 원래 데이터로 고쳐 두라고 말한 뒤 자리를 떠나 버렸다.
영훈이는 최대한 빠르게 그래프 데이터를 복원해야 한다. 하지만 자신이 어디에 어떤 수를 썼는지 전혀 기억하지 못하는 영훈이는 고민에 빠졌다.
영훈이는 자신이 데이터에 한 일은 수를 임의의 위치에 넣은 것뿐이라는 것을 알기에, 데이터에서 몇 개의 수를 지워 그래프 데이터를 만들려 한다. 즉, 수들의 원래 순서를 유지하면서 수 일부를 지워, 남은 수들이 '올바른 데이터' 가 되도록 하려는 것이다. 이때 '올바른 데이터' 란,
- 4개 이상의 정수로 이루어져 있고,
- 택희가 처음 만든 데이터의 형식을 따르며,
- 택희가 지키려 했던 모든 조건을 지키고 있는 데이터를 의미한다.
영훈이는 그래프를 복원할 방법이 여러 가지라면 기왕이면 노드의 수(N)가 가장 큰 것을, 그러한 것도 여러 가지라면 간선의 수(M)가 가장 큰 것을 복원할 것이다.
현재 데이터의 상태가 주어졌을 때, 영훈이가 복원할 그래프의 노드의 수와 간선의 수를 알아내보도록 하자.
문제풀이
(0) 생각
(1) 코드 25~38
(2) 코드 7~24
(3) 코드 52~67
#include <bits/stdc++.h> | |
using namespace std; | |
const int MAX = 1 << 18; | |
int K, arr[MAX]; | |
int segtree[2 * MAX]; | |
void update(int x, int num) { | |
segtree[x] = num; | |
while (x / 2 > 0) { | |
x /= 2; | |
segtree[x] = max(segtree[2 * x], segtree[2 * x + 1]); | |
} | |
} | |
// 0이상 ~ To미만 가장 큰 수 | |
int Maxi(int To, int idx = 1, int s = 0, int e = MAX - 1) { | |
if (e < To) return segtree[idx]; | |
if (s >= To) return 0; | |
int mid = (s + e) / 2; | |
int ret = max(Maxi(To, 2 * idx, s, mid), Maxi(To, 2 * idx + 1, mid + 1, e)); | |
return ret; | |
} | |
int numtree[2 * MAX]; // num[MAX+x] : x 값을 갖는 애들 수 | |
void num_update(int x) { | |
while (x > 0) { | |
numtree[x]++; | |
x /= 2; | |
} | |
} | |
// num번째로 작은 수 찾기 | |
int Mini(int num, int idx = 1, int s = 0, int e = MAX - 1) { | |
if (idx >= MAX) return s; | |
int mid = (s + e) / 2; | |
if (numtree[2 * idx] >= num) return Mini(num, 2 * idx, s, mid); | |
else return Mini(num - numtree[2 * idx], 2 * idx + 1, mid + 1, e); | |
} | |
int main() {ios_base::sync_with_stdio(false); cout.tie(NULL); cin.tie(NULL); | |
cin >> K; | |
for (int i = 0; i < K; ++i) { | |
cin >> arr[i]; | |
update(MAX + i, arr[i]); | |
} | |
int ans_N = -1; | |
int ans_M = -1; | |
for (int i = K - 1; i > 0; i--) { // k-1에서 1까지 (0 전까지) 볼꺼임 | |
int now = arr[i]; // i번째 수 | |
int need = 2 * now; | |
if (numtree[1] >= need) { | |
int befo = Maxi(i); // M 앞의 가장 큰 수 | |
int afto = Mini(need); // M 뒤의 2*M번째로 작은 수 | |
if (befo >= afto) { | |
if ((ans_N < befo) || (ans_N == befo && ans_M < now)) { | |
ans_N = befo; | |
ans_M = now; | |
} | |
} | |
} | |
num_update(MAX + now); | |
} | |
if (ans_N == -1) cout << -1 << '\n'; | |
else cout << ans_N << " " << ans_M << '\n'; | |
return 0; | |
} |
댓글
댓글 쓰기
긴 글 읽어주셔서 감사합니다.
궁금한게 있으시다면 댓글 달아주세요!