Doby's Lab

[알고리즘] 백준 11561번: 징검다리 (C++), 등차수열의 합, 이젠 수학까지 해야 하니 본문

PS/BOJ

[알고리즘] 백준 11561번: 징검다리 (C++), 등차수열의 합, 이젠 수학까지 해야 하니

도비(Doby) 2021. 11. 15. 17:41

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

 

11561번: 징검다리

각 테스트 케이스마다 한 줄에 승택이가 밟을 수 있는 최대 징검다리 수를 출력한다.

www.acmicpc.net

이번 문제는 수학 카테고리에 넣고 싶었으나 그래도 문제의 베이스는 이분 탐색이라서 이 카테고리에 넣어둔다.

점프를 최대로 하려면 1칸 점프를 한 뒤, +1 점프, +1 점프로 목적지에 도달해야 최댓값이 도출될 수 있다.

 

하지만, 100칸이 있을 때는 1 + 2 + 3 + 4 + 5 + 6 + 7 + 8 + 9 + 10 + 11 + 12 + 13 + 14까지 더하면 100이 넘는다.

그래서 이럴 경우에는 13번째 점프를 할 때, 100을 채우게끔 22칸을 뛰어주면 된다.

즉, 코드로 구현을 할 때는 value가 입력 값으로 들어오면 저 값이 넘지 않는 최댓값을 구해주면 된다.


솔루션

value로 들어올 수 있는 값의 범위가 엄청 크기 때문에 이때 이분 탐색을 사용해야 한다.

하지만, 시간 초과가 났었다 이유는 합을 구하는 과정에서 더하기 연산이 무수히 많이 일어나기 때문이다.

 

그래서 이번 문제의 키워드는 그 합을 어떻게 시간을 줄여서 구할 것인지가 중요하다.

>> 등차수열의 합 사용

이번 문제의 핵심 키워드는 2개였다.

1) 최댓값은 1부터 시작하는 차수가 1인 등차수열을 구해야 한다는 점

2) 등차수열의 합을 시간적인 측면에서 공식을 사용하여 빠르게 구한다는 점

#include <iostream>
#include <algorithm>
using namespace std;
#define ll unsigned long long

int T;
// n이 주어졌을 때, n부터 1까지 더한 값이 value를 넘으면 X
// value보다 작거나 같으면 O

bool sol(ll mid, ll value) {
	// 등차수열의 합
    // 소수점이 사라질까 걱정 안 해도 되는 이유는
    // +1을 해버리면 둘 중에 무조건 하나는 짝수이기 때문이다.
	ll sum = mid * (mid + 1) / 2;
	return sum > value;
}
int main() {
	cin >> T;
	ll value;
	for (int i = 0; i < T; i++) {
		cin >> value;

		ll low = 1;
		ll high = value;
		ll answer = 0;
		while (low <= high) {
			ll mid = (low + high) / 2;

			if (sol(mid, value)) {
				high = mid - 1;
			}
			else {
				low = mid + 1;
				answer = mid;
			}
		}
		cout << answer << '\n';
	}
	return 0;
}
728x90