알고리즘/[ 개념 ]

[ 개념 ] 44. 이분 매칭(Bipartite Matching)

kim.svadoz 2020. 9. 17. 18:14
반응형

> 이분매칭

번에 올릴 글은 네트워크 플로우 개념의 연장...은 아닌 것 같고,

유량 그래프의 아주 특수하면서 메이저한 형태 하나를 다룰 겁니다.

- BOJ[2188] : 축사배정

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

 

2188번: 축사 배정

농부 존은 소 축사를 완성하였다. 축사 환경을 쾌적하게 유지하기 위해서, 존은 축사를 M개의 칸으로 구분하고, 한 칸에는 최대 한 마리의 소만 들어가게 계획했다. 첫 주에는 소를 임의 배정해�

www.acmicpc.net

저번 글에서 축사 배정 문제를 다뤘는데요.

image-20200917172430877

이 문제의 유량 그래프 형태가 이러했고, 이런 형태가 상당히 특수하고 자주 나타난다고 말했는데, (모든 간선 용량은 1입니다)

여기서 왼쪽 열의 정점들은 모두 소스에서 갈 수 있고, 오른쪽 열의 정점들은 모두 싱크로 갈 수 있으며,

그 외의 간선은 모두 왼쪽 열에서 오른쪽 열로 가는 것들 뿐입니다. 이 그래프를 좀 바꾸면...

image-20200917172443508

이런 형태로 축약하는 것이 가능합니다.

정점을 두 개의 그룹으로 나누었을 때, 존재하는 모든 간선의 양 끝 정점이 서로 다른 그룹에 속하는 형태의 그래프를 이분 그래프(bipartite graph)라고 하는데요.

이분 그래프에서 한쪽 그룹을 A, 다른 쪽 그룹을 B라 할 때

아까의 그림처럼 소스에서 각 A 정점으로 가는 간선, 각 B 정점에서 싱크로 가는 간선들이 추가되고(이 간선들의 용량은 모두 1) A, B 그룹 사이의 간선의 방향은 모두 A->B일 때

최대 유량을 구하는 문제를 이분 매칭(bipartite matching) 문제라고 부릅니다.

여기서 이분 매칭 문제의 답은, 축약된 이분 그래프에서 매칭(matching)의 최대 개수와 같습니다. 이걸 최대 매칭(maximum matching)이라 칭하기도 합니다.

매칭은 간선 하나를 선택하는 것인데, 이때 간선을 선택하면서 양 끝 정점도 같이 선택하는 것이나 마찬가지고, 각 정점은 한 번까지만 선택될 수 있습니다.

따라서 위 그래프에서 간선 (A, 2), (A, 5)를 동시에 선택하는, 혹은 매칭시키는 것은 불가능합니다. A를 두 번 선택하게 되니까요.

각 정점을 한 번까지만 선택할 수 있는 조건은 소스, 싱크와 이어진 간선 용량이 모두 1이라는 점에서 구현됩니다.

지금 이분 매칭 문제를 표현하고 있는 이분 그래프에서 그려지지 않은 부분은 어차피 항상 같은 형태이기 때문에 생략한 겁니다.

이분 그래프만이 갖는 아주 단순한 구조와 거기서 파생되는 특성들 때문에, 최대 O(VE) 시간으로 빠르게 최대 매칭 개수를 구하는 최적화가 존재합니다. 여기서 V는 양쪽 그룹 중 큰 쪽의 크기라 합시다.

그냥 플로우 그래프로 두고 에드몬드 카프 알고리즘을 돌리면 min(O(VE^2), O(Ef))인데, f가 최대 O(V)이므로 O(VE)의 시간복잡도가 발생하고 이와 동일합니다.

이분 그래프에서 왼쪽 그룹을 A, 오른쪽 그룹을 B라고 합시다.

포드 풀커슨이건 에드몬드 카프건, 일단 알고리즘을 돌리면 소스 s에서 시작하는 경로를 찾기 위해 첫 번째로 방문하는 노드는 A에 속하는 정점 중 하나일 겁니다. 이걸 a라 합시다.

그리고 a에서 방문 가능한 정점은 반드시 B에 속합니다. 이걸 b라 합시다.

b에서는 일단은 싱크 t로밖에 갈 수 없습니다. 기본적으로 경로는 s->a->b->t 꼴입니다.

그러나 b에서 음의 유량이 흐르는 경로를 찾아서 A에 속하는 다른 정점으로 갈 수도 있겠죠.

이게 반복되면 s->a->b->a'->b'->t, s->a->b->a'->b'->a''->b''->t, ... 식의 경로들이 발생 가능합니다. A, B에 속하는 정점들이 반드시 교차되어 나타납니다.

양쪽 끝의 s, t는 어차피 정해져 있으니까 생략해 버리고, 중간 과정을 최대한 간소화하자는 것이 아이디어입니다.

image-20200917172456019

정점은 A 그룹의 것을 순서대로 훑으면서 아직 매칭이 안 된 애들이 있으면 매칭을 시도합니다.

또한 인접 리스트의 원소들(B 그룹에 속함) 역시 위부터 아래로 정렬되어 있다고 합시다.

맨 처음엔 (A, 2) 매칭이 바로 이어집니다.

image-20200917172506716

그 다음, B의 인접 원소가 {2, 3, 4}인데 그 중 2가 이미 누군가와 매칭이 된 상태고 그 상대가 A인데,

A를 2 대신 다른 정점과 매칭시킬 수 있나 탐색했더니 가능합니다. 5와 매칭시킬 수 있네요.

그래서 매칭 (A, 2)가 소거되고 새로운 매칭 (B, 2), (A, 5)가 생성됩니다.

이때, 매칭을 하는 데 성공했다는 말은 반드시 현재까지에 이어 매칭 수를 1 증가시켰다는 말입니다.

만약 이런 식으로 원래 매칭 상대이던 애를 다른 애와 매칭시키는 식으로 짝을 다시 지어주는 것이 불가능하면 실패입니다.

image-20200917172540695

그 다음, C는 그냥 바로 1과 잇습니다.

image-20200917172552132

D... 이놈이 1을 건드리니까 (C, 1)이 소거되고 C를 5와 매칭시키고,

역시 원래 매칭이던 (A, 5)가 사라지고 A는 다른 짝 2를 찾았으며, 그에 따라 (B, 2)도 사라지고...

그래도 다행히 B가 마지막으로 매칭 (B, 3)을 찾아서 매칭 재배치는 성공했습니다.

E의 경우, 매칭을 더 만들 수가 없습니다.

일단 유일한 인접원소가 2인데, 2와 매칭되어 있던 A를 대신 5와 매칭시키고,

5와 매칭되어 있던 C를 1과, 1과 매칭되어 있던 D를 2 또는 5와 매칭시켜야 하는데,

2는 이번에 이미 E와 매칭되어 있으므로 더 이상 매칭을 시킬 수 없고,

5 역시 이미 A와 매칭되어 있으므로 더 이상 매칭을 시킬 수 없습니다.

따라서 dead end가 발생하고 새로운 매칭을 형성시킬 수 없습니다.

만약 E의 인접원소가 더 있었다면 다른 원소와 매칭 시도를 해 볼 수도 있었겠지만, 그것조차 실패할 수도 있습니다.

이렇게 최종적인 답, 최대 매칭은 4입니다.

#include <cstdio>
#include <vector>
#include <algorithm>
using namespace std;
const int MAX = 200;

// N: A 그룹 크기, M: B 그룹 크기
// A[i], B[i]: 각 정점이 매칭된 반대편 정점 번호
int N, M, A[MAX], B[MAX];
// adj[i]: A[i]와 인접한 그룹 B의 정점들
vector<int> adj[MAX];
bool visited[MAX];

// A그룹에 속한 정점 a를 이분 매칭시켜서 성공하면 true
bool dfs(int a){
    visited[a] = true;
    for(int b: adj[a]){
        // 반대편이 매칭되지 않았거나
        // 매칭되어 있었지만 원래 매칭되어 있던 정점을 다른 정점과 매칭시킬 수 있으면 성공
        if(B[b] == -1 || !visited[B[b]] && dfs(B[b])){
            A[a] = b;
            B[b] = a;
            return true;
        }
    }
    // 매칭 실패
    return false;
}

int main(){
    scanf("%d %d", &N, &M);
    for(int i=0; i<N; i++){
        int S;
        scanf("%d", &S);
        for(int j=0; j<S; j++){
            int k;
            scanf("%d", &k);
            adj[i].push_back(k-1);
        }
    }

    int match = 0;
    // 초기값: -1
    fill(A, A+N, -1);
    fill(B, B+M, -1);
    for(int i=0; i<N; i++){
        // 아직 매칭되지 않은 그룹 A 정점에 대해 매칭 시도
        if(A[i] == -1){
            // visited 배열 초기화
            fill(visited, visited+N, false);
            if(dfs(i)) match++;
        }
    }
    printf("%d\n", match);
}

소스 코드는 상당히 짧은 편입니다.

- BOJ[9576] : 책 나눠주기

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

 

9576번: 책 나눠주기

백준이는 방 청소를 하면서 필요 없는 전공 서적을 사람들에게 나눠주려고 한다. 나눠줄 책을 모아보니 총 N권이었다. 책이 너무 많기 때문에 백준이는 책을 구분하기 위해 각각 1부터 N까지의 ��

www.acmicpc.net

비슷한 문제는 엄청나게 많습니다.

이 문제의 경우는 각 학생마다 선택지가 구간으로 주어졌을 뿐, 그 외는 축사 배정 문제와 동일합니다.

다만 이 문제의 경우 그래프가 완전 이분 그래프에 가까울수록 시간이 많이 걸릴 것인데,

그렇게 빡빡한 그래프는 주어지지 않는 것으로 보입니다. 단순한 이분 매칭으로 충분히 풀립니다.

- BOJ[11375] : 열혈강호

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

 

11375번: 열혈강호

강호네 회사에는 직원이 N명이 있고, 해야할 일이 M개가 있다. 직원은 1번부터 N번까지 번호가 매겨져 있고, 일은 1번부터 M번까지 번호가 매겨져 있다. 각 직원은 한 개의 일만 할 수 있고, 각각��

www.acmicpc.net

이 문제 또한 축사 배정 문제와 완전히 똑같습니다.

이렇게 이분 매칭의 문제 구조 자체가 엄청나게 단순하다 보니, 그냥 정말 뻔하게 뭘 서로 매칭시키는 문제는 풀기가 쉽습니다.

그래서 이분 매칭 문제들은 약간 형태가 변형되거나, 자신이 이분 매칭 문제임을 어느 정도 숨기기도 합니다.

-BOJ[11376] : 열혈강호 2

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

 

11376번: 열혈강호 2

강호네 회사에는 직원이 N명이 있고, 해야할 일이 M개가 있다. 직원은 1번부터 N번까지 번호가 매겨져 있고, 일은 1번부터 M번까지 번호가 매겨져 있다. 각 직원은 최대 두 개의 일을 할 수 있고, �

www.acmicpc.net

이 문제의 경우, 각 직원이 최대 한 개가 아니라 두 개의 일을 할 수 있다고 합니다.

여태까지는 모든 정점은 매칭에 한 번만 포함될 수 있었는데, 이 규칙이 깨지다니?? 어떻게 해야 하죠?

답은 의외로 간단합니다. 각 직원마다 정점을 2개씩 만드는 겁니다.

image-20200917172731099image-20200917172745558

예제 그래프입니다. 답은 4가 됩니다

- BOJ[11377] : 열혈강호 3

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

 

11377번: 열혈강호 3

첫째 줄에 직원의 수 N과 일의 개수 M, 일을 2개할 수 있는 직원의 수 K가 주어진다. (1 ≤ N, M ≤ 1,000, 1 ≤ K ≤ N) 둘째 줄부터 N개의 줄의 i번째 줄에는 i번 직원이 할 수 있는 일의 개수와 할 수 있

www.acmicpc.net

이 문제는 좀 더 트리키합니다. N명 중 K명이 최대 2개의 일을 할 수 있는데, 그 K명이 누구인지는 상관없습니다.

제일 처음에 생각해볼 수 있는 방법은 열혈강호 2 문제처럼 각 직원마다 정점을 2개로 분리하고, 유량을 흘리다가 N+K개가 되면 멈추는 것입니다.

그러나 이 경우, 만약 N=5, K=3일 때 1, 2, 3, 4번 직원이 2개씩 일을 해서 유량이 8이 될 수도 있는데 이때는 2개의 일을 하고 있는 직원이 4명이므로 잘못된 경우가 됩니다. 이래서는 안 됩니다.

각 직원을 2개의 정점으로 분할하였을 때, 첫 번째 정점 그룹을 P, 두 번째 정점 그룹을 Q라 합시다.

일단 맨 처음에는 P 그룹의 N개 정점들에 대해서만 최대 매칭을 구합니다.

그 다음, Q 그룹의 N개 정점들이 대해서만 또다시 최대 매칭을 구하는데, 이때 Q 그룹에서 매칭이 K개 발생하면 중단합니다. 그때의 총 매칭 개수가 답입니다.

이것이 성립하는 이유는, Q 그룹에서 매칭을 새로 찾다가 원래 매칭이 있던 P 그룹의 정점이 매칭이 없어지는 일이 절대 없기 때문입니다. 이건 보통의 이분 매칭 그래프에서도 성립합니다. 있던 매칭을 끊고 다른 매칭을 찾는 데 성공했으면, 원래 있던 매칭의 양 끝 정점도 마지막엔 새로운 매칭에 포함될 수밖에 없죠.

따라서 Q 그룹에서 매칭이 K개가 발생하는 순간 중단하면, 일을 2개 하는 직원이 K명 이하라는 조건을 지킬 수 있습니다.

이제부터는 자기가 이분 매칭이 아니다! 라고 위장하고 있는 문제들이나, 좀 어려운 문제들입니다.

- BOJ[3295] : 단방향 링크 네트워크**

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

 

3295번: 단방향 링크 네트워크

첫째 줄에 테스트 케이스의 개수 T가 주어진다. 각 테스트 케이스의 첫째 줄에는 상호 네트워크를 이루는 노드의 수 n과 단방향 링크의 수 m이 주어진다. (n ≤ 1,000, m ≤ 50,000) 노드는 0번부터 n-1��

www.acmicpc.net

이 문제의 경우, 각 정점을 다 2개로 나눠서 이분 매칭을 시도하면 되는데 도대체 왜냐면,

이 문제에서 하라는 행동을 잘 살펴보면, 겹치지 않는 링과 선형 배열이 만족하는 공통점이 있습니다.

바로, 어떤 정점이 있다면 절대 indegree나 outdegree가 2 이상일 수 없다는 겁니다.

따라서 모든 정점을 나가기만 하는 정점 그룹 A, 들어오기만 하는 정점 그룹 B로 분리하고,

모든 방향 간선 (u, v)에 대해 u가 A, v가 B에 속한다고 하면 모든 정점이 매칭에 2번 이상 선택되지 않는 최대 매칭과 동치가 됩니다.

- BOJ[1671] : 상어의 저녁식사

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

 

1671번: 상어의 저녁식사

어떤 상어는 저녁식사로 서로를 먹는다. 모든 상어는 자신과 다른 상어의 크기, 속도, 지능을 수치로 나타낸 것을 알고 있다. 만약, 상어 A의 크기, 속도, 지능이 상어 B의 크기, 속도, 지능보다 크

www.acmicpc.net

이 문제도 상당히 유사합니다. 상어를 잡아먹는 쪽 정점과 잡아먹히는 쪽 정점으로 분류하면,

(일단은 최대 2마리까지 잡아먹을 수 있다는 건 보류합니다)

image-20200917173508372

예제를 이렇게 나타낼 수 있습니다. 매칭이 의미하는 것은 왼쪽 정점이 오른쪽 정점을 먹었다는 겁니다.

image-20200917173742727

이 두 매칭은 2번 상어가 4번 상어를 먹고, 또 4번 상어가 1번 상어를 먹었다는 의미인데요.

이 매칭의 순서는 상관이 없습니다. 4번이 잡아먹기도, 잡아먹히기도 하는데,

문제에서 구하고자 하는 게 살아남는 상어의 최소 마릿수이기 때문에 잡아먹을 수 있는 한 최대한 많이 잡아먹게 만드는 게 목표입니다. 그러고 남은 상어 마릿수를 세는 거죠.

따라서 이 경우, 만약 4번이 1번을 먹기 전에 2번이 4번을 먹어버리면 1번 상어가 살아남을 수도 있지만(최대 2마리만 먹을 수 있다는 조건 때문에),

우리의 목적을 이루기 위해 이럴 때는 무조건 약한 놈이 먼저 더 약한 상어를 먹어치우게 하는 순서로 가야 합니다. 즉, 1번이 4번에게 먹힌 후 4번이 2번에게 먹혀야 합니다.

따라서 이렇게 연쇄적인 매칭이 일어나도 이렇게 순서를 어떻게든 재배열하여 매칭에 속한 오른쪽의 모든 상어가 잡아먹히게 만들 수 있습니다.

또한 각 상어가 최대 2마리까지 먹을 수 있다는 조건을 위해, 왼쪽 정점을 열혈강호 2 문제처럼 2개로 분열시키면 됩니다.

마지막으로 주의해야 하는 예외는 완전히 능력치가 동일한 상어가 여러 마리 존재할 때입니다. 이 경우는 잘못하면 서로가 매칭되어서 둘 다 사라져버릴수도;

이때는 무슨 순서를 매겨서 순서가 앞인 놈만 다른 놈들을 잡아먹을 수 있도록 예외처리를 해야 합니다. 뭐... 입력받는 순서가 제일 적당하겠죠.

- BOJ[9525] : 룩 배치하기

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

 

9525번: 룩 배치하기

체스와 관련된 문제는 처음 알고리즘을 배우기 시작할 때 부터 접하게 된다. 가장 유명한 문제로는 N-Queen 문제가 있다. 이를 변형해서 N-Rook 문제를 만들 수 있다. Rook(룩) 은 체스에서 같은 행이

www.acmicpc.net

이 문제도 살펴봅시다. 이제부터 좀 더 굉장합니다.

폰이나 벽이 나오기 전까지의 자신의 상하좌우 빈칸 모두로 이동할 수 있는 룩들을 최대한 많이 배치할 건데, 이들이 서로 이동경로가 하나도 겹치면 안 됩니다.

일단 처음 알 수 있는 어찌보면 당연한 성질은, 하나의 일렬로 된 빈칸 행 중에서는 최대 한 군데에만 룩을 배치할 수 있습니다. 이건 열 또한 마찬가지죠.

그런데, 어떤 칸에 룩을 배치한다고 해봅시다. 그렇다면 이 칸을 포함하는 일렬로 된 빈칸 행과, 일렬로 된 빈칸 열에는 더 이상 룩을 배치할 수 없습니다.

이걸 잘 변형시켜 보면, 각 일렬로 된 빈칸들이 정점일 때, 어떤 칸에 룩을 배치하는 것이 자신을 포함하는 두 빈칸 정점을 매칭하는 것이고, 각 빈칸 정점들은 최대 1개의 매칭만을 가질 수 있게 됩니다.

image-20200917174430416

이렇게 예제의 맵에서 정점들을 추출할 겁니다.

파란색이 A 그룹, 연속한 빈 공간 행이고, 빨간색이 B 그룹, 연속한 빈 공간 열입니다.

이제 겹치는 영역끼리 간선을 서로 추가해 주고, 그 간선은 모두 딱 하나의 칸에 대응될 겁니다.

그 간선을 뽑아 매칭시키는 것은 그 대응하는 칸에 룩을 배치하는 것이죠.

image-20200917174446427

그 이분 그래프는 이런 형태가 됩니다!! 두둥!!

- BOJ[1348] : 주차장

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

 

1348번: 주차장

세준 주차장은 R*C크기의 직사각형 모양이다. 세준 주차장에는 N개의 차와, M개의 주차 구역이 있다. 그리고, 모든 차는 주차 구역에 주차하려고 한다. 교통 제한 때문에, 차는 주차장의 경계와 평

www.acmicpc.net

이 문제는 또 어떨까요? 꽤나 어려운 문제 중 하나입니다.

일단 이분 매칭이 정해라는 걸 안다는 가정하에, 어떻게 이분 매칭을 도입할 수 있을지 봅시다.

당연히 존재하는 게 차와 주차 구역이니 각각을 매칭시키는 것이 답이겠지요.

각 주차 구역에는 차가 한 대밖에 들어갈 수 없으니까요.

이때 문제는 모든 차가 주차하는 데 걸리는 최소 시간입니다.

최소 시간을 구하기 전에, k초 안에 모든 차가 주차 가능한지를 판단할 수 있을까요?

이때는 차와 주차 구역을 이을 때, 둘 사이의 거리가 k 이하일 때만 잇습니다.

이렇게 이분 그래프를 만든 후 최대 매칭이 차의 개수이면 가능하다는 것이겠죠.

그렇습니다. k를 이분 탐색으로 찾는 겁니다.

모든 차와 주차 구역 사이의 거리는 BFS로 전처리하여 빠르게 구해둡니다.

이분 매칭 문제를 여러 개 살펴봤는데, 처음엔 쉬운 것 같으면서도 나중엔 굉장히 당황스러운 문제들이 많습니다.

역시, 기본은 쉬운데 이녀석도 어려워지면 충분히 어려워질 수가 있네요.

아, 이분 매칭의 시간복잡도가 O(VE)였죠. 가끔 이 시간복잡도로 시간 안에 해결할 수 없는 거대한 이분 그래프가 주어지는 문제들이 간혹 있는데요.

이건 역시 나중에 이분 그래프에 대해서 최대 매칭을 좀 더 빠르게 찾아주는 알고리즘과 함께 살펴보도록 합시다.

이분매칭 관련 추천 문제

2188번: 축사 배정

위에서 설명한 문제입니다.

11375번: 열혈강호

위에서 설명한 문제입니다.

9576번: 책 나눠주기

위에서 설명한 문제입니다.

1298번: 노트북의 주인을 찾아서

이 또한 주는 문제입니다.

11376번: 열혈강호 2

위에서 설명한 문제입니다.

11377번: 열혈강호 3

위에서 설명한 문제입니다.

1017번: 소수 쌍 (★)

유명한 문제입니다. 모든 자연수들을 홀수 그룹짝수 그룹으로 나눕니다.

1이 두 번 존재하지는 않으므로, 서로 다른 두 수를 더해서 나올 수 있는 소수인 수는 반드시 홀수여야만 하고, 그러려면 반드시 홀수와 짝수를 더해야 합니다.

소름 끼친다;

따라서 홀수 그룹 정점 u와 짝수 그룹 정점 v가 있을 때 u+v가 소수일 때 둘 사이에 간선을 추가한 후 이분 매칭입니다.

이때, 첫 번째 수와 가능한 모든 매칭을 다 해 보면서, 각각 최대 매칭이 N/2인지 판별하면 됩니다.

N이 홀수일 수도 있는 것으로 압니다. 이러면 당연히 걍 불가능.

9577번: 토렌트 (★)

정점으로 만들어야 할 것은 조각단위시간입니다.

1

n번 조각이 그룹 A, 0

1, 1

2, 2

3, ... 등의 단위시간이 그룹 B에 속합니다.

이제 각 조각마다 받을 수 있는 단위시간과 연결하여 이분 그래프를 만든 후,

가능한 한 가장 빠른 시간 정점부터 방문해가며 이분 매칭을 시도합니다.

image-20200917174802578

입력

3 2

1 3 1 1

0 2 2 2 3

의 예시입니다. 오른쪽이 시간 정점인데, k는 k-1~k초를 뜻합니다.

마지막에 최대 매칭이 N 미만이면 불가능합니다.

가능하다면, 매칭 정점 배열을 참조하여 가장 마지막으로 무언가를 받은 시간을 출력하면 됩니다.

3295번: 단방향 링크 네트워크 (★)

위에서 설명한 문제입니다.

1671번: 상어의 저녁식사 (★)

위에서 설명한 문제입니다.

1574번: 룩 어택

룩 배치하기 문제보다 쉽습니다. 이번엔 폰이 아니라 빈칸이 주어지는데, 빈칸엔 룩을 배치할 수 없다 뿐이지 건너편에도 룩이 없어야 합니다.

따라서 정점이 항상 행 개수와 열 개수라, 정점을 일일이 만드는 난잡한 전처리가 필요하지 않아서 더 쉽습니다.

9525번: 룩 배치하기 (★)

위에서 설명한 문제입니다.

2570번: 비숍 2 (★)

비숍은 대각선으로만 움직일 수 있습니다. 방향만 다를 뿐 룩 배치하기 문제와 완전히 같습니다.

다만 전처리 과정이 좀 더 머리 아파질 듯...

1348번: 주차장 (★)

위에서 설명한 문제입니다. 이분 탐색입니다.

 

**참조**

blog.naver.com/kks227

반응형