bjfu1109 最小公倍数和

这题真是过了n年才a。最早是在2010年北大培训比赛上看到的这题,当时我不会,竹教主也不会,但他记下来了,研究一段时间后就会了,还把这题加到我校oj上。过了这么多年,我上网搜,关于这个问题的解题报告还是没有,于是我花了几天时间做了出来,发布此解题报告。

题目是要求从1到n的所有数与n的最小公倍数的和,再除了n。第一眼看到数据范围,就知道是不能硬做(即从1到n依次算一遍)的。那么,为了找规律,得把公式化一化。

f(n) = [lcm(1,n)+lcm(2,n)+……+lcm(n,n)]/n
   = 1/gcd(1,n)+2/gcd(2,n)+……+n/gcd(n,n)

于是f(n)化成了一堆数的和。

显然,每一个gcd(k,n)都是能整除n的,按gcd(k,n)的值进行分类合并,就可以把f(n)整理成以下形式

f(n) = (……)/1 + (……)/2 +……+ (……)/d

这里d就是所有能整除n的整数。而(……)/1的分子为所有与n互质的数的和;(……)/2的分子为所有与n的最大公约数为2,也就是与n/2互质的数的和;依次类推,得到公式

有了这个公式,虽然是前进了一大步,但依然没法解题。因为这还是必须枚举n的所有约数,而这个复杂度依然为O(n)。

接下来就是比较核心的一步,构造。

f(n) = (g(n) + 1) / 2,求得g(n)就立即可得f(n)了。

这里构造g(n)的原因,是因为g(n)有积性性质。

对于任意gcd(m, n) = 1,有g(m * n) = g(m) * g(n)。证明略掉,读者自证。

有了积性性质就好办了,将n质因数分解成pi^ci相乘的形式,n = ∏(pi^ci),则g(n) = g(∏(pi^ci)) = ∏(g(pi^ci))

g(pi^ci)很好计算。这里的d就是1, pi, pi^2, pi^3, ..., pi^ci,而phi(pi^ci)=(pi-1)*pi^(ci-1),合起来就是个等比数列,可以推出来一个公式。也就是说可以O(1)时间算出g(pi^ci)。

如此一来,整个问题就可解了。

不过这题还是很变态的,因为数据就有50000组,仅分解质因数写得不好,都可能挂掉。最后我还是过了,代码如下:

/*
 * bjfu1109
 * Author    : ben
 */
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <ctime>
#include <iostream>
#include <algorithm>
#include <queue>
#include <set>
#include <map>
#include <stack>
#include <string>
#include <vector>
#include <deque>
#include <list>
#include <functional>
#include <numeric>
#include <cctype>
using namespace std;
#ifdef ON_LOCAL_DEBUG
#else
#endif
typedef long long LL;
typedef int typec;

LL getPow(int a, int b) {
    LL res, temp;
    res = 1, temp = (LL) a;
    while (b) {
        if (b & 1) {
            res = res * temp;
        }
        b >>= 1;
        temp = temp * temp;
    }
    return res;
}

int get_int() {
    int res = 0, ch;
    while (!((ch = getchar()) >= ‘0‘ && ch <= ‘9‘)) {
        if (ch == EOF)
            return -1;
    }
    res = ch - ‘0‘;
    while ((ch = getchar()) >= ‘0‘ && ch <= ‘9‘)
        res = res * 10 + (ch - ‘0‘);
    return res;
}

const int N = 100000;
bool isPrime[N + 3];//多用两个元素以免判断边界
vector<int> pt;
void init_prime_table() {
    memset(isPrime, true, sizeof(isPrime));
    int p = 2, q, del;
    double temp;
    while (p <= N) {
        while (!isPrime[p]) {        p++;        }
        if (p > N) {//已经结束
            break;        }
        temp = (double) p;
        temp *= p;
        if (temp > N)
            break;
        while (temp <= N) {
            del = (int) temp;            isPrime[del] = false;
            temp *= p;        }
        q = p + 1;
        while (q < N) {
            while (!isPrime[q]) {    q++;    }
            if (q >= N) { break;}
            temp = (double) p;
            temp *= q;
            if (temp > N)    break;
            while (temp <= N) {
                del = (int) temp;
                isPrime[del] = false;
                temp *= p;
            }
            q++;
        }
        p++;
    }
    for (int i = 2; i <= N; i++) {
        if (isPrime[i]) {
            pt.push_back(i);
        }
    }
}

/**
 * p为素数表,至少应该包括sqrt(N)以内的所有素数,f保存结果
 * f[i].first表示N的一个素因数,f[i].second为这个素因子的个数
 * typec可以是int、long、long long等
 */
typedef vector<pair<typec, int> > FactorList;
void get_prime_factor(const typec &N_, FactorList &f, const vector<int> &p) {
    int i, t, n, pl = p.size();
    typec N = N_;
    f.clear();
    for(i = 0; i < pl; i++) {
        t = p[i];
        if (t * t > N_) {
            break;
        }
        if(N % t == 0) {
            n = 0;
            while(N % t == 0) {
                n++;    N /= t;
            }
            f.push_back(make_pair(t, n));
        }
        if(N == 1) {    break;    }
    }
    if(N > 1) {
        f.push_back(make_pair(N, 1));
    }
}

inline LL g(LL p, int c) {
    LL ret = getPow(p, 2 * c) - 1;
    ret = ret / (p + 1) * p + 1;
    return ret;
}

FactorList fl;
LL g(int n) {
    get_prime_factor(n, fl, pt);
    int len = fl.size();
    LL ret = 1;
    for (int i = 0; i < len; i++) {
        ret *= g(fl[i].first, fl[i].second);
    }
    return ret;
}

int main() {
#ifdef ON_LOCAL_DEBUG
    freopen("data.in", "r", stdin);
#endif
    LL ans, N;
    init_prime_table();
    while ((N = get_int()) > 0) {
        ans = g(N) + 1;
        ans /= 2;
        printf("%I64d\n", ans);
    }
    return 0;
}
时间: 2024-11-10 07:32:42

bjfu1109 最小公倍数和的相关文章

HDU 1108: 最小公倍数

最小公倍数 #include<iostream> using namespace std; int gcd(int a, int b) { while (b) { int t = a%b; a = b; b = t; } return a; } int lcm(int a, int b) { return a / gcd(a, b)*b; } int main() { ios::sync_with_stdio(false); int a, b; while (cin >> a &g

辗转相除法 求最大公约数和最小公倍数

# include<stdio.h> int main() { int a,b,c,x,y; printf("请输入两个正整数,用逗号间隔:"); scanf("%d,%d",&a,&b); x=a; y=b; if (a<b) { c=a; a=b; //要保证 a>b b=c; } while (b!=0) { c=a; a=b; b=c%b; } c=x*y/a; printf("最大公约数为%d,最小公倍数为

luoguP1029 最大公约数和最小公倍数问题 [gcd][数论]

题目描述 输入二个正整数x0,y0(2<=x0<100000,2<=y0<=1000000),求出满足下列条件的P,Q的个数 条件: 1.P,Q是正整数 2.要求P,Q以x0为最大公约数,以y0为最小公倍数. 试求:满足条件的所有可能的两个正整数的个数. 输入输出格式 输入格式: 二个正整数x0,y0 输出格式: 一个数,表示求出满足条件的P,Q的个数 输入输出样例 输入样例#1: 3 60 输出样例#1: 4 说明 P,Q有4种 3 60 15 12 12 15 60 3 先考虑

poj3101--Astronomy(分数的最小公倍数)

题目链接:点击打开链接 题目大意:有n个行星,给出每个行星的旋转的周期.问最少多少时间后n个行星会在一条直线上,初始点在一起,不存在全部的行星都有同一个周期 如果A行星的周期是t1.B行星的周期是t2(t2>t1),要在一条直线上,一定会执行的相差半个周期的倍数,时间(t/t2 - t/t1) % (1/2) = 0.也就是t*(t1-t2)/(t1*t2)%(1/2) = 0,要是时间最小.所以也就是差出一个半周期.也就是t = (t2-t1)/(t2*t1*2)这个t也就是A.B执行到一条直

最大公约数和最小公倍数

输入两个正整数m和n求最大公约数和最小公倍数. #include <stdio.h> int main() { int m, n; printf("请输入你要计算的两个数:"); scanf("%d %d", &m, &n); printf("最大公约数是:"); printf("%d\n", get_max_common_measure(m, n)); printf("最小公倍数是:&q

欧几里德公式求最大公因数和最小公倍数

如下: #include<stdio.h> int gcd(int a,int b){ int temp; if(b > a){ temp = b; b = a; a = temp; } while(b){ temp = a%b; a = b; b = temp; } return a; } int lcm(int a,int b,int gcd){ int temp; if(b > a){ temp = b; b = a; a = temp; } return (a/gcd)*b

写一个方法,求两个数的最大公约数和最小公倍数。

package homework0702; /* * 最大公约数 利用辗转相除法求解两个正整数的最大公约数 在循环中,只要除数不等于0,用较大的数除以较小的数,将小的一个数作为下一轮循环的大数,取得的余数作为下一轮循环较小的数,如此循环直到较小的数值为0,返回较大的数.即为最大公约数. 辗转相除法(欧几里得算法) 定理:两个整数的最大公约数等于其中较小的那个数和两数的相除余数的最大公约数.最大公约数(greatest common divisor)缩写为gcd. 最小公倍数 最小公倍数 = (a

算法训练 最大最小公倍数

时间限制:1.0s   内存限制:256.0MB 问题描述 已知一个正整数N,问从1~N中任选出三个数,他们的最小公倍数最大可以为多少. 输入格式 输入一个正整数N. 输出格式 输出一个整数,表示你找到的最小公倍数. 样例输入 9 样例输出 504 数据规模与约定 1 <= N <= 106. 解题思路:其实看似挺简单的一道题,但是却需要分析一番.虽然知道是用贪心法,下面的代码能ac,但自己的贪心策略远远不够准确,不如最后面的方法简洁准确.自己的分析能力还是很烂. #include <i

洛谷 P1029 最大公约数和最小公倍数问题 Label:Water&amp;&amp;非学习区警告

题目描述 输入二个正整数x0,y0(2<=x0<100000,2<=y0<=1000000),求出满足下列条件的P,Q的个数 条件: 1.P,Q是正整数 2.要求P,Q以x0为最大公约数,以y0为最小公倍数. 试求:满足条件的所有可能的两个正整数的个数. 输入输出格式 输入格式: 二个正整数x0,y0 输出格式: 一个数,表示求出满足条件的P,Q的个数 输入输出样例 输入样例#1: 3 60 输出样例#1: 4 说明 P,Q有4种 3 60 15 12 12 15 60 3 代码