mayoko’s diary

プロコンとかいろいろ。

Japan Alumni Group Summer Camp 2015 Day 2 B - 監獄

解法

二分探索をします。

ok(x) = (x 以下の数のうち, N 回目の操作で選ばれる数があるか) という判定をします。この判定ができると, 結局のところ求めたい x というのは ok(x) が true になるような数の中で最小の数, ということになります。これは二分探索でいけそうです。

では ok(x) を求めるプログラムを考えます。

1 回の操作では, x 以下の数のうち, 0 及び K の倍数が消されます。よって, x 以下の数のうち, 残る数の個数は x/K + 1 個減ります。このようにして x 以下の数のうち最大のものが 0 番目以上の数として残れば ok(x) は true です。

int N, K;

bool ok(ll x) {
    for (int i = 0; i < N; i++) {
        if (x >= 0 && i == N-1) return true;
        x -= x/K + 1;
    }
    return false;
}

int main() {
    cin.tie(0);
    ios::sync_with_stdio(false);
    cin >> N >> K;
    ll low = -1, high = 1ll<<60;
    while (high - low > 1) {
        ll mid = (low+high) / 2;
        if (ok(mid)) high = mid;
        else low = mid;
    }
    cout << high << endl;
    return 0;
}