codeforces 300E Empire Strikes Back 数论+二分查找

题意:给定N个数a1,a2,a3...aN,现在要求最小的n满足 n!/(a1!*a2!*...*aN!) 是一个正整数的最小的n。

分析:这题的想法很明确,就是分解a1!*a2!*...*aN!,把其分解成质因子相乘的形式,这个都很熟悉了,然后就是对每一个质因子二分搜索出一个数字下界,最后求其中最大的一个数,问题的关键就是如何分解这样一个表达式成一个质因子相乘的形式。使用一个cnt数组来表示每一个数的在乘积中出现的次数,然后从后往前假设一个数出现了k次,那么如果这个数是素数则不用更新,如果一个数是合数则将其分解成两部分,一个是该数最小的质因子,一个是除以这个质因子之后的值,接着一直做下去,就能够把所有的素因子全部统计起来,最后再对每一个素因子都二分搜索。

#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <algorithm>
#include <set>
#include <cmath>
#include <vector>
using namespace std;

typedef long long LL;
const int N = 10000005;
vector<int>vv;
LL cnt[N];
int p[N];
int Max;
LL sum;
int n;

void pre() {
    for (int i = 2; i < N; ++i) {
        if (!p[i]) {
            p[i] = i;
            vv.push_back(i);
        }
        for (int j = 0; i*vv[j] < N; ++j) {
            p[i*vv[j]] = vv[j];
            if (i % vv[j] == 0) break;
        }
    }
}

LL cal(LL mid, LL base) {
    LL ret = 0;
    while (mid) {
        ret += (mid /= base);
    }
    return ret;
}

void deal() {
    for (int i = Max; i >= 2; --i) {
        if (p[i] != i) {
            cnt[p[i]] += cnt[i];
            cnt[i/p[i]] += cnt[i];
        }
    }
}

LL get(LL base, LL x) {
    LL l = 1, r = sum;
    LL ret;
    while (l <= r) {
        LL mid = (l + r) >> 1;
        if (cal(mid, base) >= x) {
            ret = mid;
            r = mid - 1;
        } else {
            l = mid + 1;
        }
    }
    return ret;
}

int main() {
    pre();
    scanf("%d", &n);
    int x;
    for (int i = 0; i < n; ++i) {
        scanf("%d", &x);
        sum += x;
        Max = max(Max, x);
        ++cnt[x];
    }
    for (int i = Max-1; i >= 2; --i) {
        cnt[i] += cnt[i+1];
    } // 模拟阶乘,1-n之间每个数都有一个 
    deal();
    LL ret = 1;
    for (int i = 0; i < vv.size(); ++i) {
        ret = max(ret, get(vv[i], cnt[vv[i]]));
    }
    printf("%I64d\n", ret);
    return 0;
} 

 

你可能感兴趣的:(codeforces)