출처: https://3months.tistory.com/307 [Deep Play]

백준/Greedy

[BOJ] 19580 마스크가 필요해

코딩하는 랄뚜기 2022. 2. 23. 20:23

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

 

19580번: 마스크가 필요해

첫 번째 줄에는 A도시의 시민 수인 N과 A도시의 상점 수인 M이 주어진다. (1 ≤ N, M ≤ 500,000) 두 번째 줄부터 N + 1번째 줄까지는 A도시의 각 시민이 마스크에 소비할 수 있는 돈의 범위인 Li, Ri가

www.acmicpc.net


문제

코로나19가 발생하고 나서 마스크의 필요성이 증가하기 시작했다. 마스크가 없으면 생활을 하는데 어려움이 있기 때문에 마스크를 항상 구비하고 있어야 한다.

마스크의 수요가 증가함에 따라 일정했던 마스크의 가격이 상점마다 달라지게 되었다. 마스크의 가격이 제각각이기 때문에 A도시의 시민들이 전부 마스크를 갖기가 어려워지기 시작했다. A도시의 공무원인 당신은 이 사태를 해결하기 위해서 상점의 주인들에게 마스크의 가격을 일정하게 해달라고 했지만, 상점 주인들은 당연히 무시했다.

상점 주인들을 설득시키는 것은 어렵다고 생각해서 당신은 최대한 많은 시민들이 마스크를 가질 수 있게 하는 것으로 계획을 변경했다. 당신은 A도시의 각 시민이 마스크에 소비할 수 있는 금액의 범위와 A도시의 상점에서 팔고 있는 마스크의 가격 및 개수를 알아냈다. 각 시민은 최대 1개의 마스크만 살 수 있다. 이 정보를 바탕으로 최대한 많은 시민들이 마스크를 얻을 수 있게 하자.

입력

첫 번째 줄에는 A도시의 시민 수인 N과 A도시의 상점 수인 M이 주어진다. (1 ≤ N, M ≤ 500,000)

두 번째 줄부터 N + 1번째 줄까지는 A도시의 각 시민이 마스크에 소비할 수 있는 돈의 범위인 Li, Ri가 주어진다. 즉 i번째 A도시 시민이 살 수 있는 마스크의 가격은 Li 이상 Ri 이하이다. (1 ≤ Li ≤ Ri ≤ 1018)

N + 2번째 줄부터 N + M + 1번째 줄까지는 A도시의 각 상점이 판매하고 있는 마스크의 가격인 Pj와 마스크의 개수인 Xj가 주어진다. (1 ≤ Pj ≤ 1018, 1 ≤ Xj ≤ 1,000)

모든 Li, Ri, Pj, Xj는 정수이다.

출력

가능한 한 많은 시민이 마스크를 샀을 때, 마스크를 산 시민의 수를 출력한다.


풀이

  1. 마스크는 사람이 구매할 수 있는 최소비용을, 버스는 마스크 가격을 기준으로 오름차순 정렬한다.
  2. 마스크를 기준으로 반복문을 돌면서 사람이 구매할 수 있는 최소비용이 마스크 가격을 넘지 않는다면 우선순위 큐에 넣어준다. 구매할 수 있는 최대 비용이 작은 것에 우선순위를 줘서 넣어줘야 한다.
  3. 해당 마스크의 개수만큼 반복문을 돌면서 우선순위 큐를 pop한다만약 최대 비용이 마스크 가격보다 크거나 같으면 정답+1을 해준다.
  4. 2~3번을 반복한다.

코드

#include <string>
#include <stdio.h>
#include <math.h>
#include <iostream>
#include <vector>
#include <algorithm>
#include <map>
#include <queue>
#include <stack>
#include <climits>

#define SWAP(a, b, type) do { \
    type temp; \
    temp = a;  \
    a = b;     \
    b = temp;  \
} while (0)

#define INF 1000000000
#define endl '\n'
#define ll long long

using namespace std;

int N,M;

struct cmp1{
    bool operator()(pair<ll,ll> a,pair<ll,ll> b){
        return a.second>b.second;
    }
};

vector<pair<ll,ll> > p;
vector<pair<ll,ll> > m;
priority_queue<pair<ll,ll>, vector<pair<ll,ll> >, cmp1 > pq;


void input() {
    cin>>N>>M;
    ll a,b;
    for(int i=0;i<N;i++){
        cin>>a>>b;
        p.push_back(make_pair(a,b));
    }
    for(int i=0;i<M;i++){
        cin>>a>>b;
        m.push_back(make_pair(a,b));
    }
}

void init() {}


void solution() {
    ll ans=0;
    ll a,b,cost,num;
    sort(p.begin(),p.end());
    sort(m.begin(),m.end());
    
    int idx=0;
    for(int i=0;i<M;i++){
        cost=m[i].first, num=m[i].second;
        
        //가능한 최소비용 넣어주기
        while(idx<N&&p[idx].first<=cost){
            pq.push(make_pair(p[idx].first,p[idx].second));
            idx++;
        }
        
        //최대비용을 빼면서 허용가능한 범위 안에 들면 ans++
        while(!pq.empty()&&num){
            if(cost<=pq.top().second) ans++,num--;
            pq.pop();
        }
    }
    cout<<ans;
}

int main()
{
    ios::sync_with_stdio(0); cin.tie(0); cout.tie(0);
    input();
    solution();
    return 0;
}