일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |
- SECS-II
- 프로그래머스
- SECS/GEM
- Spotify Api
- 파이썬
- modern c++
- 회귀
- 스포티파이
- python
- 자바
- spotify
- java
- C++
- 회원가입
- CS
- regression
- Gem
- programmers
- Spring JPA
- SWEA
- SW Expert Academy
- c
- 백준
- SECS
- Spring
- MYSQL
- Computer Science
- Baekjoon
- spring boot
- linux
- Today
- Total
비버놀로지
[BAEKJOON 백준] 11000 강의실 배정 (JAVA) 본문
https://www.acmicpc.net/problem/11000
문제
수강신청의 마스터 김종혜 선생님에게 새로운 과제가 주어졌다.
김종혜 선생님한테는 Si에 시작해서 Ti에 끝나는 N개의 수업이 주어지는데, 최소의 강의실을 사용해서 모든 수업을 가능하게 해야 한다.
참고로, 수업이 끝난 직후에 다음 수업을 시작할 수 있다. (즉, Ti ≤ Sj 일 경우 i 수업과 j 수업은 같이 들을 수 있다.)
수강신청 대충한 게 찔리면, 선생님을 도와드리자!
입력
첫 번째 줄에 N이 주어진다. (1 ≤ N ≤ 200,000)
이후 N개의 줄에 Si, Ti가 주어진다. (0 ≤ Si < Ti ≤ 109)
출력
강의실의 개수를 출력하라.
예제 입력 1 복사
3
1 3
2 4
3 5
예제 출력 1 복사
2
그리디 알고리즘과 우선순위 큐를 사용하여 풀 수 있는 문제였습니다.
강의실을 여러 개 사용하되, 강의가 겹치지 않게 최소한의 강의실을 배정하는 것이 목표입니다.
N은 200,000이기 때문에 단순하게 브루트포스를 돌리면 당연히 시간초과가 나고, 똑똑한 방식을 생각해야 합니다.
먼저, 어떻게 강의를 배정해야하는지부터 고민해 봅시다.
당연히, 버리는 시간을 최소로 하면서 강의를 배정해야 최소한의 강의실의 개수를 알아낼 수 있을 것입니다.
그렇다면, 어떻게 하면 버리는 시간을 최소로 할 수 있을까요?
방법은 바로 정렬입니다.
시작 시각을 기준으로 하거나, 종료 시각을 기준으로 할 수 있는데 시작 시각을 기준으로 오름차순 정렬해야 합니다.
종료 시각을 기준으로 하면 왜 안 되는지에 대해서는 포스팅 하단에 따로 적어두겠습니다.
특정 강의를 시작 시각을 기준으로 오름차순 정렬하되, 시간 시각이 같다면 종료 시각을 기준으로 오름차순 정렬한다면 아래와 같은 형태가 될 것입니다.
여기서 버려지는 시간을 색칠해 봅시다.
빨간색으로 칠한 영역이 강의를 배정할 수 없는 버려지는 시간입니다.
아무렇게나 강의실을 넣는 것보다 시작 시각으로 정렬을 해야 앞에서부터 차곡 차곡 강의 간의 텀을 짧게 하여 배정이 가능합니다.
이렇게 문제의 입력으로 들어온 강의 배열을 시작 시각을 기준으로 정렬이 끝났다면, 강의실이 몇 개가 필요한지 생각해봐야 합니다.
위의 그림에서 알 수 있듯이, 정렬 자체는 시작 시각을 기준으로 하였지만, 결국 어떤 강의가 들어갈 자리를 찾기 위해서는 현재 배정된 강의의 종료 시각과 비교하게 됩니다.
가령, A 강의가 1시부터 3시까지 진행하고, B 강의가 2시부터 4시까지 진행한다면, A 강의의 종료 시각이 3시고, B 강의의 시작 시각이 2시이므로 B 강의는 새로운 강의실에 배정해야 합니다.
그리고 항상 배정된 강의의 시작 시각보다 빠른 강의는 없으므로 강의가 겹치는 일도 생기지 않습니다.
위 로직을 구현하기 위해서는 우선순위 큐가 제격입니다.
우선순위 큐에 강의의 종료 시각을 넣어 두고, 정렬 기준은 오름차순으로 설정합니다.
그리고 배정이 되지 않은 강의의 시작 시각과 우선순위 큐 안에서 종료 시각이 가장 빠른 강의의 종료 시각과 비교합니다.
만약, 배정이 되지 않은 강의의 시작 시각이 우선순위 큐 안에서 종료 시각이 가장 빠른 강의의 종료 시각보다 늦다면, 현재 우선순위 큐에서 top에 있는 강의를 제거하고, 이 배정이 안 된 강의를 우선순위 큐에 넣는 것입니다.
이를, 다시 말하면 원래 있던 강의실에 다음 강의를 배정하는 것이죠.
반대의 경우는 우선순위 큐에서 top에 있는 강의는 냅두고, 배정이 안 된 강의를 우선순위 큐에 넣어 줍니다.
이것은 새로운 강의실에 강의를 배정하는 것과 일치합니다.
그리고 모든 강의가 배정이 되었다면, 우선순위 큐의 사이즈를 반환하면 됩니다.
아래는 위 과정을 정리한 소스코드입니다.
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.Arrays;
import java.util.PriorityQueue;
import java.util.StringTokenizer;
public class Main {
private static class Point{
int start,end;
public Point(int start, int end) {
super();
this.start = start;
this.end = end;
}
}
public static void main(String[] args) throws NumberFormatException, IOException {
BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
int N=Integer.parseInt(br.readLine());
Point study[]=new Point[N];
StringTokenizer st=null;
for (int i = 0; i < N; i++) {
st=new StringTokenizer(br.readLine());
study[i]=new Point(Integer.parseInt(st.nextToken()),Integer.parseInt(st.nextToken()));
}
// 시작 시간을 기준으로 오름차순 정렬하되,
// 시작 시간이 같다면, 종료 시간을 기준으로 오름차순 정렬한다.
Arrays.sort(study, (l1, l2) -> l1.start == l2.start ? l1.end - l2.end : l1.start - l2.start);
PriorityQueue<Integer> pq = new PriorityQueue<>();
pq.offer(study[0].end);
for (int i = 1; i < N; i++) {
// 우선순위 큐에서 가장 작은 종료 시간과
// 현재 lectures[i]의 시작 시간을 비교한다.
if (pq.peek() <= study[i].start) {
pq.poll();
}
pq.offer(study[i].end);
}
System.out.println(pq.size());
}
}
'ALGORITM > JAVA' 카테고리의 다른 글
[BAEKJOON 백준] 10282 해킹 (JAVA) (0) | 2022.05.20 |
---|---|
[BAEKJOON 백준] 1655 가운데를 말해요 (JAVA) (0) | 2022.05.16 |
[BAEKJOON 백준] 2306 유전자 (JAVA) (0) | 2022.05.14 |
[BAEKJOON 백준] 14462 소가 길을 건너간 이유 8 (JAVA) (0) | 2022.05.14 |
[BAEKJOON 백준] 5875 오타 (JAVA) (0) | 2022.05.12 |