2110번: 공유기 설치
첫째 줄에 집의 개수 N (2 ≤ N ≤ 200,000)과 공유기의 개수 C (2 ≤ C ≤ N)이 하나 이상의 빈 칸을 사이에 두고 주어진다. 둘째 줄부터 N개의 줄에는 집의 좌표를 나타내는 xi (0 ≤ xi ≤ 1,000,000,000)가
www.acmicpc.net
문제제목 : 공유기 설치
문제유형 : 이분탐색
문제난이도 : 실버1
문제
도현이의 집 N개가 수직선 위에 있다. 각각의 집의 좌표는 x1, ..., xN이고, 집 여러개가 같은 좌표를 가지는 일은 없다.
도현이는 언제 어디서나 와이파이를 즐기기 위해서 집에 공유기 C개를 설치하려고 한다. 최대한 많은 곳에서 와이파이를 사용하려고 하기 때문에, 한 집에는 공유기를 하나만 설치할 수 있고, 가장 인접한 두 공유기 사이의 거리를 가능한 크게 하여 설치하려고 한다.
C개의 공유기를 N개의 집에 적당히 설치해서, 가장 인접한 두 공유기 사이의 거리를 최대로 하는 프로그램을 작성하시오.
입력
첫째 줄에 집의 개수 N (2 ≤ N ≤ 200,000)과 공유기의 개수 C (2 ≤ C ≤ N)이 하나 이상의 빈 칸을 사이에 두고 주어진다. 둘째 줄부터 N개의 줄에는 집의 좌표를 나타내는 xi (0 ≤ xi ≤ 1,000,000,000)가 한 줄에 하나씩 주어진다.
출력
첫째 줄에 가장 인접한 두 공유기 사이의 최대 거리를 출력한다.
아이디어
문제를 읽었을 때, 문제가 이해가 안돼서 힘들었던 문제이다...
해당 문제에서 주어진 TC에서,
[1 2 - 4 - - - 8 - 9 ] 일때 출력이 3이라는데, (1,4,9)에 설치해서 4와 9사이는 5인데.. 하면서 뭔 소린가 한참 해맸었다.
결국 블로그를 참고하게 되었는데, 최대 거리가 4이면 (1,8)을 선택할 수 밖에 없으니 안되고
최대거리가 3이면 (1,4,8)도 되고, (1,4,9)도 가능하게 된다.
최대 거리가 2면 (1,4,8), (1,4,9)이렇게 공유기 3개 설치가 가능한데, 최대거리를 구하는 것이므로 3을 출력해준 것이다.
따라서, 공유기를 설치할 수 있는 집의 최소 거리는 1이고
최대 거리는 (가장 먼 집 - 가장 가까운 집)이 된다.
최소 거리부터 최대거리까지 포문을 돌리는 것 보다 이분탐색을 이용하여 탐색하는 것이 효율적이므로
이분탐색을 이용하면 된다.
package week6;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.util.Arrays;
public class BOJ2110 {
public static int N;
public static int C;
public static int[] distance;
public static void main(String[] args) throws Exception {
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
String[] input = br.readLine().split(" ");
N = Integer.parseInt(input[0]);
C = Integer.parseInt(input[1]);
distance = new int[N];
for (int i = 0; i < N; i++) {
distance[i] = Integer.parseInt(br.readLine());
}
Arrays.sort(distance);
int min = 1; // 최소 간격
int max = distance[N-1] - distance[0]; // 최대간격
int result = 0; // 결과 (최대 거리)
int last_set = 0; // 가장 마지막에 공유기를 설치한 집
while (min <= max) {
last_set = distance[0];
int count = 1; // 공유기 개수 1 : 첫번째 집에는 설치
int mid = (max + min) / 2;
for (int i = 1; i < distance.length; i++) {
if (distance[i] - last_set >= mid) {
last_set = distance[i];
count++;
}
}
if (count >= C) {
result = Math.max(result, mid);
min = mid + 1;
} else { // 해당 거리로 공유기를 설치할 경우, 공유기가 남아서 탐색할 거리를 좁혀준다.
max = mid - 1;
}
}
System.out.println(result);
}
}
어제 밤에 풀다가 도저히 모르겠어서 문제만 이해하고 오늘 다시 도전해봤는데 은근 쉽게 풀린 문제.
빨리 실버문제는 껌으로 풀면서 골드문제로 넘어가고 싶다 ㅠ_ㅠ... 알고리즘 어려워
'코딩테스트 준비 > JAVA 코테' 카테고리의 다른 글
백준 1747. 소수&팰린드롬 (브루트포스, 소수, 팰린드롬) (0) | 2021.04.07 |
---|---|
백준 7568. 덩치 (0) | 2021.04.07 |
백준 1654. 랜선 자르기 (0) | 2021.04.05 |
백준 10816. 숫자 카드 2 (0) | 2021.04.05 |
백준 13305. 주유소 (0) | 2021.04.01 |
댓글