题意:给出n(1<=n<=10^8),求小于n的,求所有与n互质的数字的四次幂的加和是多少。

分析:容斥原理

首先要知道四次幂求和公式,1^4+2^4+...+n^4=n*(n+1)*(2n+1)*(3n^2+3n-1)/30

先求所有小于等于n的数字的四次幂和,然后减去那些不互质的即可。

这个减去的过程用到了容斥原理。

先对n分解质因子,每个不同的质因子只保留一个。

然后分别枚举这些质因子的组合情况,由奇数个因子组成的数要减去,由偶数个因子组成的数要加上。

对于一个因子组合的乘积a,我们需要一次性计算a^4+(2a)^4 + (3a)^4+...

将其转化为a^4 * (1^4+2^4+...)即可。

 

这道题还有一个难点,就是公式中有除法(除以30),却还要进行模运算。

除法是不支持模运算的,因此我们要将除法转化为乘法,除以30变为乘以30的逆元。

逆元的意思是,如果a、b互为mod c下的逆元,则a * b = 1 (mod c)。

求逆元可以用扩展欧几里德gcd(30,MOD,x,y),把x/gcd(30,MOD)整理到0~MOD-1范围内即为30的逆元。

具体原因查阅扩展欧几里德算法

zju3547_i++zju3547_质因子_02
#include <cstdio>
using namespace std;

#define D(x) 

const int MOD = (int)(1e9) + 7;
const int MAX_FACTOR = 40;

int n;
int factor_num;
long long factor[MAX_FACTOR];
long long inverse;

//n(n+1)(2n+1)(3n^2+3n-1)/30

long long to_forth(long long value)
{
    long long ret = value;
    ret = ret * ret % MOD;
    ret = ret * ret % MOD;
    return ret;
}

long long cal(long long value)
{
    long long num = n / value;
    long long ret = 1;
    ret = ret * num % MOD * (num + 1) % MOD;
    ret = ret * (2 * num + 1) % MOD;
    ret = ret * ((num * num % MOD * 3 % MOD + 3 * num % MOD - 1) % MOD) % MOD;
    if (ret / 30 != ret * inverse % MOD)
    {
        D(printf("#%lld %lld\n", ret / 30, ret * inverse % MOD));
    }else
    {
        D(printf("**\n"));
    }
    ret = ret * inverse % MOD;

    ret = ret * to_forth(value) % MOD;

    return ret;
}

void get_factors()
{
    factor_num = 0;
    int m = n;
    for (int i = 2; i * i <= m; i++)
    {
        if (m % i == 0)
            factor[factor_num++] = i;
        while (m % i == 0)
        {
            m /= i;
        }
    }
    if (m != 1)
    {
        factor[factor_num++] = m;
    }
}

long long work()
{
    long long ans = 0;
    for (int i = 1; i < (1 << factor_num); i++)
    {
        int num = 0;
        long long temp = 1;
        int index = 0;
        for (int mask = 1; mask <= i; mask <<= 1, index++)
        {
            if ((mask & i) == 0)
            {
                continue;
            }
            num++;
            temp *= factor[index];
        }
        D(printf("temp=%lld\n", temp));
        if (num & 1)
            ans += cal(temp);
        else
            ans -= cal(temp);
        ans = (ans % MOD + MOD) % MOD;
    }
    ans = ((cal(1) - ans) % MOD + MOD) % MOD;
    return ans;
}

void gcd_extend(long long a,long long b,long long &g,long long &x,long long &y)
{
    if (!b)
    {
        g = a;
        x = 1;
        y = 0;
        return;
    }
    gcd_extend(b, a % b, g, y, x);
    y -= a / b * x;
}

int main()
{
    long long x, y, g;
    gcd_extend(30, MOD, g, x, y);
    D(printf("%lld %lld %lld\n", x, y, g));
    x = (x % MOD + MOD) % MOD;
    inverse = x / g;
    D(printf("%lld\n", inverse));
    D(printf("%lld\n", inverse * 30 % MOD));
    int t;
    scanf("%d", &t);
    while (t--)
    {
        scanf("%d", &n);
        if (n == 1)
        {
            puts("0");
            continue;
        }
        get_factors();
        int ans_int = work();
        printf("%d\n", ans_int);
    }
    return 0;
}
View Code