티스토리 뷰

728x90

문제 링크

www.acmicpc.net/problem/10800

 

10800번: 컬러볼

첫 줄에는 공의 개수를 나타내는 자연수 N이 주어진다(1 ≤ N ≤ 200,000). 다음 N개의 줄 중 i번째 줄에는 i번째 공의 색을 나타내는 자연수 Ci와 그 크기를 나타내는 자연수 Si가 주어진다(1 ≤ Ci ≤ N

www.acmicpc.net

 

풀이

구현 문제라 사실상 코드 설명이 되겠습니다.

 

1) 모든 공들을 크기순으로 정렬해줍니다. (코드상 $p$ 벡터)

2) 각 색별로 공들의 사이즈를 오름차순으로 저장해줍니다. (코드상 $cnt$ 벡터)

3) 모든 공을 순회하며, 자신보다 작은 사이즈 공 중(이전에 순회한 공들), 같은 색의 공을 제외한 나머지 사이즈합을 정답에 추가해줍시다.

 

정답 코드

#include <bits/stdc++.h>

#define INF 0x3f3f3f3f
#define ft first
#define sd second
#define all(x) (x).begin(), (x).end()
using namespace std;
using pii = pair<int, int>;
using vpii = vector<pii>;

int n;
vector<tuple<int, int, int>> p;
vpii cnt[222'222];
int ans[222'222];

int main() {
    ios_base::sync_with_stdio(false);
    cin.tie(nullptr);

    cin >> n;
    for (int i = 0; i < n; i++) {
        int u, v;
        cin >> u >> v;
        p.emplace_back(v, u, i);
        cnt[u].emplace_back(v, v);
    }
    for (int i = 1; i < n + 1; i++) {
        cnt[i].emplace_back(0, 0);
        sort(all(cnt[i]));
        for (int j = 1; j < cnt[i].size(); j++) {
            cnt[i][j].sd += cnt[i][j - 1].sd;
        }
    }

    sort(all(p));
    int sum = 0, res = get<0>(p[0]);
    for (int i = 1; i < p.size(); i++) {
        auto[s, c, idx] = p[i];
        if (get<0>(p[i - 1]) != s) {
            sum += res;
            res = 0;
        }
        int cntIdx = upper_bound(all(cnt[c]), make_pair(s - 1, INF)) - cnt[c].begin() - 1;
        ans[idx] = sum - cnt[c][cntIdx].sd;
        res += s;
    }
    for (int i = 0; i < n; i++) {
        cout << ans[i] << '\n';
    }
}
728x90

'알고리즘 > 문제 풀이' 카테고리의 다른 글

[BOJ] 백준 2306 - 유전자  (0) 2021.03.30
[BOJ] 백준 12886 - 돌 그룹  (0) 2021.03.30
[BOJ] 백준 1424 - 새 앨범  (0) 2021.03.13
[BOJ] 백준 1670 - 정상 회담 2  (0) 2021.03.13
[BOJ] 백준 2186 - 문자판  (0) 2021.03.13
댓글
공지사항
최근에 올라온 글
최근에 달린 댓글
Total
Today
Yesterday
«   2024/05   »
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 29 30 31
글 보관함