알고리즘/BOJ

[BOJ 8394, C++] 악수

70825 2022. 8. 9. 12:34
반응형

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

 

8394번: 악수

첫째 줄에 회의에 참석한 사람의 수 n (1 ≤ n ≤ 10,000,000)이 주어진다.

www.acmicpc.net

 

 

1. 문제 증명


예시가 4만 나와있는데, n = 1 ~ 3일 때도 확인해보면 피보나치 수열의 점화식이 나옵니다.

arr[i] = arr[i-1] + arr[i-2]

 

피보나치 수열 점화식을 증명하는 방법은 생각보다 간단합니다.

회의에 참석한 사람 N명이 있고, 이미 우리가 1명부터 N-1명일 때까지의 경우의 수를 모두 구했다고 가정해보겠습니다.

이때 N-1명일 때의 경우의 수는 dp[N-1]입니다.

 

N번째 사람이 추가되면 경우의 수가 2가지로 나뉘게 됩니다.

 

1) N번째 사람이 악수를 하지 않는 경우

이 경우엔 1번째 사람 ~ N-1번째 사람이 악수한 경우의 수를 구하면 되므로 dp[N-1]이 됩니다.

 

2) N번째 사람이 악수를 한 경우

N번째 사람이 악수를 하는 경우는 N-1번째 사람과 무조건 악수를 하는 경우뿐입니다.

그래서 N-1번째 사람과 N번째 사람은 악수를 하는 것이 확정되었으므로 dp[N-2]를 구하면 됩니다.

 

증명 내용 자체는 수학적 귀납법까지 사용해야 완벽하지만, 문제 푸는데 그정도는 필요한 내용이 아니라서 생략하겠습니다.

 

그래서 결국에는 dp[N] = dp[N-1] + dp[N-2]가 나오게 됩니다.

보통 이런 문제를 보면 예시를 구해보고, 규칙성을 찾아 바로 코드로 짜서 제출했는데, 생각해보니 그동안 따로 피보나치 수열 점화식에 대한 증명을 해보지 않았어서 한 번 해봤네요.


 

2. 코드


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
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
 
const int N = 10000002;
 
ll dp[N] = { 0 };
 
 
int main()
{
    std::ios::sync_with_stdio(false);
    cin.tie(NULL);
    memset(dp, -1sizeof(dp));
 
    int n;
    cin >> n;
 
    dp[1= 1; dp[2= 2;
    for (int i = 3; i <= n; i++) {
        dp[i] = (dp[i - 1+ dp[i - 2]) % 10;
    }
    
    cout << dp[n];
 
    return 0;
}
cs

 

반응형