HDU5528 - Count a * b

HDU5528 - Count a * b


做法:求\(\sum_{m|n}(m^2 - \sum_{i=1}^{m}\sum_{j=1}^m [m|(ij)])\)

\(h(m) = \sum_{i=1}^{m}\sum_{j=1}^m [m|(ij)] = \sum_{i=1}^m\sum_{j=1}^m [\frac{m}{(i,m)}|\frac{i}{(i,m)}j]\)
$ = \sum_{i=1}^m\sum_{j=1}^m [\frac{m}{(i,m)}|j] = \sum_{i=1}^m \frac{m}{\frac{m}{(i,m)}} = \sum_{i=1}^m (i,m)$
$ = \sum_{d|m} d \sum_{i=1}^m [(i,m)=d] = \sum_{d|m} d \sum_{i=1}^{\frac{m}{d}} [(i,\frac{m}{d})=1] = \sum_{d|m}d\varphi(\frac{m}{d})$

$
\sum_{m|n} \sum_{d|m}d\varphi(\frac{m}{d}) = \sum_{d|n}d\sum_{m|\frac{n}{d}}\varphi(m) = \sum_{d|n}d\frac{n}{d} = \sum_{d|n}n
$
所以\(ans = \sum_{d|n} d^2 - n\sigma_0(n)\)两个值均为积性函数,预处理素因子,合并答案即可。本题时限较紧,不能直接使用试除法分解,尽量优化常数。

#include <bits/stdc++.h>
typedef unsigned long long ll;
const int N = 1e6 + 7;
inline int read() {
    char c = getchar(); int x = 0, f = 1;
    while( ! isdigit(c) ) { if(c == '-') f = -1; c = getchar(); }
    while( isdigit(c) ) { x = x*10 + c - '0'; c = getchar(); }
    return x * f;
}
inline void write( ll x ) {
    if( x >= 10 ) write( x / 10 );
    putchar( x % 10 + '0' );
}
using namespace std;
int T;
ll n;
int p[N], notp[N];
void init() {
    notp[1] = 1;
    for(int i = 2; i <= 1e6; ++i) {
        if(!notp[i]) p[++p[0]] = i;
        for(int j = 1; j <= p[0] && p[j]*i <= 1e6; ++j) {
            notp[i*p[j]] = 1;
            if(i%p[j] == 0) break;
        }
    }
}
ll solve(ll n) {
    ll ans = 1, num = 1, tn = n;
    for(int i = 1; i <= p[0] && p[i]*p[i] <= n; ++i) if(n%p[i] == 0){
        ll t = p[i], tmp1 = 1, tmp2 = 0;
        while(n%t==0) {
            n/=t;
            tmp1 = tmp1*t*t + 1;
            ++tmp2;
        }
        ans *= tmp1;
        num *= (tmp2 + 1);
    }
    if(n!=1) {
        num*=2;
        ans *= (1 + n*n);
    }
    ans -= num*tn;
    return ans;
}
int main() {
    T = read();
    init();
    while(T--) {
        n = read();
        write(solve(n)); putchar('\n');
    }
    return 0;
}

转载于:https://www.cnblogs.com/RRRR-wys/p/9736250.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值