Notice
Recent Posts
Recent Comments
Link
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |
Tags
- 세그먼트 트리
- 16985
- NCP
- gorilla/mux
- 누적합
- golang
- 구현
- DP
- 다익스트라
- 시뮬레이션
- 점수 따먹기
- redis
- BOJ
- 21921
- 11659
- Naver Cloud
- 백준
- 크루스칼
- 맥주 축제
- 최소신장트리
- mst
- SWEA
- 구간합
- 17503
- 정렬
- 민준이와 마산 그리고 건우
- mongodb
- 이분 탐색
- dfs
- c++
Archives
- Today
- Total
Gi-Log
백준(BOJ) 1654 랜선 자르기 C++ 풀이 본문
문제 링크: https://www.acmicpc.net/problem/1654
문제 풀이에 이용된 알고리즘: 이분 탐색
주어진 서로 다른 길이의 k개의 랜선을 똑같은 길이로 토막을 내어서, 동일한 길이의 랜선 n개를 만드는 가장 긴 랜선 토막의 길이를 찾는 문제이다.
랜선 토막의 길이를 1부터 1씩 증가시켜 나가면, 조건을 만족하는 가장 긴 토막의 길이를 구하는 것은 굉장히 단순하지만 확실한 방법이라는 것은 모두가 알 것이다.
하지만 누가봐도 비효율적인 방법이며, 이런 탐색을 조금 더 효율적으로 하기 위해서 이분 탐색으로 풀이를 진행하였다.
토막의 길이가 0일 수는 없으므로 left는 1로 초기화하고, right는 현재 주어진 랜선 중 가장 긴 것의 길이로 초기화한다.
이후부터는 이분 탐색을 통해 토막의 길이를 변화시키며, 해당 토막의 길이로 랜선들을 잘랐을 때 몇개의 토막을 얻을 수 있는 지 체크하는 방식으로 루프를 돌게 된다.
파이썬이라면 상관 없겠지만 C++ 사용자라면 이 문제에서 고려해야될 점이 하나 있다. mid를 구하기 위해서 left와 right를 더하고 2로 나누어야하는데, 이 때 int 형을 사용하면 오버플로우가 발생할 수 있다.
따라서 이런 상황을 모면하기 위해서 left, right, mid를 모두 long long 타입으로 선언해주었다.
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
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
|
/* BOJ 1654 랜선 자르기 */
#define _CRT_SECURE_NO_WARNINGS
#include <iostream>
#include <queue>
#include <vector>
#include <algorithm>
#include <cstring>
#include <string>
#include <string.h>
#define endl '\n'
using namespace std;
typedef long long ll;
int k, n, arr[10001];
int main()
{
ios_base::sync_with_stdio(0);
cin.tie(0);
cout.tie(0);
//freopen("input.txt", "r", stdin);
cin >> k >> n;
for (int i = 1; i <= k; i++)
cin >> arr[i];
ll left = 1, right = *max_element(arr + 1, arr + 1 + k);
int ans = 0;
while (left <= right)
{
ll mid = (left + right) / 2;
int cnt = 0;
for (int i = 1; i <= k; i++)
cnt += (arr[i] / mid);
if (cnt < n) // 조금더 잘게 잘라야함
right = mid - 1;
else ans = mid, left = mid + 1; // 조금 더 길게 잘라봐도 무방함
}
cout << ans << endl;
return 0;
}
|
cs |
'알고리즘 BOJ' 카테고리의 다른 글
백준(BOJ) 13911 집 구하기 C++ 풀이 (0) | 2021.07.10 |
---|---|
백준(BOJ) 20542 받아쓰기 C++ 풀이 (0) | 2021.07.08 |
백준(BOJ) 14499 주사위 굴리기 C++ 풀이 (0) | 2021.06.29 |
백준(BOJ) 1182 부분 수열의 합 C++ 풀이 (0) | 2021.06.29 |
백준(BOJ) 14621 나만 안되는 연애 C++ 풀이 (0) | 2021.06.27 |
Comments