문제 링크
https://www.acmicpc.net/problem/1321
문제
캠프 내내 그랬듯이, 여전히 옆 나라와의 전쟁이 한창이다.
전쟁에는 N개의 부대가 투입되었는데, 전쟁이 장기전이 되다 보니 군사의 적절한 배치를 위해 각 부대에 군인이 늘어나기도 하고 줄어들기도 하고 있다.
행정의 편의를 위해 각 군인들에겐 번호가 붙어 있는데, 군인들은 1번 부대부터 군번순서대로 차례차례 배치된다. 예를 들어 1번 부대에 4명, 2번 부대에 3명, 3번 부대에 7명의 군인이 있다면 군번이 6번인 군인은 2번 부대에 배치되게 된다.
문제는 어떤 부대의 인원이 늘어나거나 줄어들었을 때 i번 군인이 어디에 배치되는지 인데, 이럴 때에는 군인도 군번도 처음부터 다시 배치하게 된다. 위의 예에서와 같이 1번 부대에 4명, 2번 부대에 3명, 3번 부대에 7명의 군인이 있었는데, 1번 부대에서 3명의 감원이 일어난다면, 6번 군인은 3번 부대에 재배치 받게 된다.
전쟁 때는 부대의 감원과 증원이 많아 군사 재배치도 자주 일어나게 되는데, 이렇게 자주 배치가 바뀌자 군인들은 자기가 도대체 어떤 부대에 속하는 지 헷갈리게 되었다. 다행히도 바뀐 군번은 다들 정확하게 숙지하고 있다.
부대의 개수 N과 각 부대에 속해 있는 군인의 수가 N개 주어질 때, 부대의 감원과 증원을 한 후, 혹은 그 중에 군번 i번의 군인이 몇 번 부대에 속하는 지를 물어봤을 때, 그 질문에 대답을 해 줄 수 있는 프로그램을 작성하시오.
i번 부대에 증원이나 감원을 할 때엔 "1 i a"의 형태로 명령이 주어지고, 이는 i번 부대에 a명을 더한다는 뜻이다. 감원을 할 때엔 a가 0보다 작은 수로 주어진다. 감원을 해서 부대의 인원수가 0보다 작아지는 입력은 들어오지 않는다. a는 절댓값이 1보다 크거나 같고, 3,000보다 작거나 같은 정수이다.
군번 i번의 군인이 어떤 부대에 배치 받았는지 알고 싶을 때는 "2 i"의 형태로 명령이 주어지고, 이런 명령을 받았을 때는 i번 군인이 몇 번 부대에 배치 받았는지를 출력해야 한다. i는 전체 군인 수보다 작거나 같은 자연수이다.
입력
첫째 줄에 부대의 개수 N(1 ≤ N ≤ 500,000)이 주어지고, 이어서 각 부대의 군사 수를 나타내는 정수가 N개 주어진다. 각 부대의 군사 수는 1000보다 작거나 같은 자연수이다. 그 다음 줄에 명령의 개수 M(1 ≤ M ≤ 10,000)개가 주어지고, 이어서 M줄에 걸쳐 명령이 주어진다.
출력
질문한 군인이 몇 번 부대에 배치 받았는지를 한 줄에 하나씩 출력한다.
예제 입력 1
5
1 4 3 5 2
5
1 2 -3
1 4 2
2 5
1 2 4
2 5
예제 출력 1
3
2
알고리즘 분류
- 자료 구조
풀이
주어진 정보를 받아, 세그먼트 트리에 각 부대에 군사 수를 저장해준다. 이후 M개의 명령을 처리한다.
우선 A가 1이라면, C의 부호에 따라 군사를 증원할 지, 뺄 지가 결정된다.
Update_SegTree(1, 1, 500000, B, C);
A가 2라면 B번 군인이 몇 번 부대에 속해 있는지를 탐색한다.
cout << Find_Value(1, 1, 500000, B) << "\n";
Find_Value() 함수를 통해 B번째 군인은 어떤 노드에 속해 있는지를 확인해준다.
코드
#include <iostream>
#include <string>
#include <cmath>
#include <set>
#include <map>
#include <stack>
#include <queue>
#include <vector>
#include <algorithm>
#include <climits>
#define FASTIO cin.tie(NULL); cout.tie(NULL); ios::sync_with_stdio(false);
#define MAX 500001
#define LL long long
#define INF 1e9
using namespace std;
int N, M;
int MAP[MAX];
vector<int> SegTree;
void Update_SegTree(int Node, int S, int E, int Val, int Diff) {
if (S == E) {
SegTree[Node] += Diff;
return;
}
int M = (S + E) / 2;
if (Val <= M) {
Update_SegTree(Node * 2, S, M, Val, Diff);
}
else {
Update_SegTree(Node * 2 + 1, M + 1, E, Val, Diff);
}
SegTree[Node] = SegTree[Node * 2] + SegTree[Node * 2 + 1];
}
int Find_Value(int Node, int S, int E, int Val) {
if (S == E) {
return S;
}
int M = (S + E) / 2;
if (Val <= SegTree[Node * 2]) {
return Find_Value(Node * 2, S, M, Val);
}
else {
return Find_Value(Node * 2 + 1, M + 1, E, Val - SegTree[Node * 2]);
}
}
void Settings() {
int Tree_Height = (int)ceil(log2(500000));
int Tree_Size = (1 << (Tree_Height + 1));
SegTree.resize(Tree_Size);
}
void Input() {
cin >> N;
for (int i = 1; i <= N; i++) {
cin >> MAP[i];
Update_SegTree(1, 1, 500000, i, MAP[i]);
}
}
void Find_Answer() {
cin >> M;
for (int i = 0; i < M; i++) {
int A, B;
cin >> A >> B;
if (A == 1) {
int C;
cin >> C;
MAP[B] += C;
Update_SegTree(1, 1, 500000, B, C);
}
else if (A == 2) {
cout << Find_Value(1, 1, 500000, B) << "\n";
}
}
}
int main() {
FASTIO
Settings();
Input();
Find_Answer();
return 0;
}
'BOJ > Platinum ~ Diamond' 카테고리의 다른 글
[BOJ/Platinum 5] 백준 4442 빌보의 생일(C++) (0) | 2022.03.02 |
---|---|
[BOJ/Platinum 5] 백준 10090 Counting Inversions(C++) (0) | 2022.03.02 |
[BOJ/Platinum 5] 백준 2243 사탕상자(C++) (0) | 2022.03.01 |
[BOJ/Platinum 5] 백준 9426 중앙값 측정(C++) (0) | 2022.03.01 |
[BOJ/Platinum 5] 백준 1572 중앙값(C++) (3) | 2022.03.01 |