알고리즘/문제풀이

[백준 31963번] 두 배

Ohnim · 오님 2024. 6. 25. 01:13

문제 링크

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

(2024 KOI 1차대회 초등부 2번, 중등부 1번)

 

문제 요약

길이가 N인 수열이 주어졌을 때, 수열의 한 원소에 숫자 2를 곱하는 연산을 할 수 있다. 이때 주어진 수열을 오름차순으로 만들 수 있는  최소한의 연산 횟수를 구하는 문제이다.

 

문제 풀이

시뮬레이션으로 문제 풀기(부분점수)

수열의 두 번째 원소부터 마지막 원소까지 순회하면서, 각 원소가 이전 원소보다 커질 때까지 2를 곱하는 과정을 반복하고, 그 횟수를 직접 계산하는 방법으로 문제를 풀 수 있다.

int simulate(const vector<int>& vi) {
    vector<int> incresing_order(vi.size());
    incresing_order[0] = vi[0];
    
    int ret = 0;
    
    for (int i = 1 ; i < vi.size() ; ++i) {
        incresing_order[i] = vi[i];
        
        while (incresing_order[i] < incresing_order[i - 1]) {
            incresing_order[i] *= 2;
            ++ret;
        }
    }
    
    return ret;
}

하지만 이 방법으로는 만점을 받을 수 없고 부분점수만 받을 수 있다. 예를 들어, 1,000,000부터 시작해 1씩 감소하는 수열이 주어졌을 때를 생각해 보자. 이 경우가 최악의 사례인데, 이때 시간복잡도는 \(\mathcal{O}(N^2)\)이므로 N이 최대일 때는 주어진 시간 내에 문제를 해결할 수 없다. 따라서 만점을 받기 위해서는 다른 방법이 필요하다.

 

수식으로 정리해서 문제 풀기

\(i-1\)번째 원소 \(A_{i-1}\)에 2가 곱해진 횟수를 \(x\)라고 하고, \(i\)번째 원소 \(A_{i}\)에 2가 곱해진 횟수를 \(y\)라고 하자. 이때 수열이 오름차순이 되려면 다음과 같은 수식을 만족해야 한다.

$$A_{i-1} \times 2^x \leq A_{i} \times 2^y$$

그 다음 \(y\)를 제외한 나머지 모든 항을 왼쪽으로 정리한다.

$$\frac{A_{i-1}}{A_{i}} \times 2^x \leq 2^y$$

이제 양쪽 항 모두에 \(log_2\)를 적용하고, 로그의 곱셈 및 나눗셈 법칙을 이용해 정리한다.

$$log_2 \frac{A_{i-1}}{A_{i}} + x \leq y$$

만약 \(i - 1\)번째 원소 \(A_{i-1}\)에 2를 곱한 횟수 \(x\)를 알고 있다면, 앞서 정리한 수식을 이용해 마지막 원소까지 2를 곱해야 하는 횟수를 모두 구할 수 있다. 첫 번째 원소에는 2를 곱하는 않는 것이 유리하므로, \(i - 1\)이 1일 때 \(x\)는 0으로 정해져 있다. 이 사실을 활용하면 문제의 답을 구할 수 있다. 답이 int 범위가 넘어갈 수 있으므로 long long 자료형을 사용해야 한다는 것에 유의하자.

long long solve(const vector<int>& vi) {
    long long ret = 0;
    
    vector<int> multiply_cnt(vi.size(), 0);
    
    for (int i = 1 ; i < vi.size() ; ++i) {
        multiply_cnt[i] = max((int)ceil(log2((double)vi[i - 1]/ (double)vi[i]) + (double)multiply_cnt[i - 1]), 0);
        
        ret += multiply_cnt[i];
    }
    
    return ret;
}
사실 위 코드는 부동소수점 연산의 정확도 문제 때문에 잠재적인 위험이 있다. 정확한 로그 값이 필요한 것이 아니라 올림(ceil) 연산을 한 값이 필요하기 때문에 vi[i - 1]과 v[i]에 직접 log2 연산을 하지 않아도 된다. 대신 vi[i - 1]에 2를 몇 번 곱해야 vi[i]보다 커지는지 혹은 vi[i - 1]에 2를 몇 번 나눠야 v[i]보다 작아지는지를 계산해 둔다면 ceil과 똑같은 기능을 기대할 수 있다. 이렇게 한다면 부동소수점 연산이 가지고 있는 부정확함에 벌벌 떨지 않아도 될 것이다.

 

소스 코드

더보기
#include <iostream>
#include <vector>
#include <algorithm>
#include <cmath>

using namespace std;

long long solve(const vector<int>& vi) {
    long long ret = 0;
    
    vector<int> multiply_cnt(vi.size(), 0);
    
    for (int i = 1 ; i < vi.size() ; ++i) {
        multiply_cnt[i] = max((int)ceil(log2((double)vi[i - 1]/ (double)vi[i]) + (double)multiply_cnt[i - 1]), 0);
        
        ret += multiply_cnt[i];
    }
    
    return ret;
}

int main() {
    int n;
    
    cin >> n;
    
    vector<int> vi(n);
    
    for (int i = 0 ; i < n ; ++i) {
        cin >> vi[i];
    }
    
    long long ans = solve(vi);
    
    cout << ans << endl;
}