zoukankan      html  css  js  c++  java
  • GuGuFishtion HDU

    GuGuFishtion

    [Time Limit: 1500 msquad Memory Limit: 65536 kB ]

    题意

    给出定义(Gu(a, b) = frac{phi(ab)}{phi(a)phi(b)})
    求出(sum_{a=1}^{m}sum_{b=1}^{n}Gu(a,b) (mod p))

    思路

    首先对于欧拉函数,我们知道欧拉函数的朴素式子为:(phi(n) = n*(1-frac{1}{p1})*(1-frac{1}{p2}) * ... * (1-frac{1}{pn}))(pi)(n) 的质因子。
    对于任意两个数 (a,b),令 (g = gcd(a, b))

    1. (g != 1),令 (pi)(a) 特有的质因子,(qi)(b) 特有的质因子,(ti)(a,b) 共有的质因子,那么将 (Gu(a, b)) 展开,就可以得到

    [egin{aligned} Gu(a, b) &= frac{phi(ab)}{phi(a)phi(b)}\ &= frac{ab prod(1-frac{1}{pi}) prod(1-frac{1}{ti}) prod(1-frac{1}{qi}) }{a prod(1-frac{1}{pi}) prod(1-frac{1}{ti}) b prod(1-frac{1}{qi})prod(1-frac{1}{ti})} \ &= frac{1}{prod(1-frac{1}{ti})} end{aligned} ]

    现在我们设 (x)(x) 包括了所有的 (ti),那么就有

    [egin{aligned} Gu(a, b) &= frac{1}{prod(1-frac{1}{ti})} \ &= frac{x}{xprod(1-frac{1}{ti})} \ &= frac{x}{phi(x)} end{aligned} ]

    (x) 也很好知道是多少,其实 (g) 就满足同时包括了所有 (ti) 的数,所以我们可以设 (x = g),就可以得到 (Gu(a,b) = frac{g}{phi(g)})
    2. 若 (g=1),此时不存在 (ti),但这是 (Gu(a, b)) 展开后全部消掉了,所以答案为 (1),而 (frac{1}{phi(1)}) 也正好为 (1),所以也可以看成 (Gu(a,b) = frac{g}{phi(g)})

    综合上述,(Gu(a,b) = frac{g}{phi(g)})
    此时我们只要计算出 (gcd(a, b) = x (ain [1,m], bin[1,n])) 的对数,就可以直接计算答案了。
    这里可以利用经典的莫比乌斯反演,也可以利用容斥原理。
    令:
    (f(i)) 表示 (gcd) 等于 (i的倍数) 的对数
    (g(i)) 表示 (gcd) 等于 (i) 的对数
    那么就有

    [f(i) = lfloorfrac{m}{i} floor lfloorfrac{n}{i} floor \ g(i) = f(i) - sum_{j=2}^{i*j<=min(n,m)} g(ij) ]

    如此倒着计算 (g(i)),就可以得出答案。

    Hint

    emmmm,这题其实有点卡常,要注意取模的次数和自然数逆元打表的姿势。

    /*************************************************************** 
        > File Name    : a.cpp
        > Author       : Jiaaaaaaaqi
        > Created Time : 2019年08月26日 星期一 16时58分58秒
     ***************************************************************/
    
    #include <map>
    #include <set>
    #include <list>
    #include <ctime>
    #include <cmath>
    #include <stack>
    #include <queue>
    #include <cfloat>
    #include <string>
    #include <vector>
    #include <cstdio>
    #include <bitset>
    #include <cstdlib>
    #include <cstring>
    #include <iostream>
    #include <algorithm>
    #include <unordered_map>
    #define  lowbit(x)  x & (-x)
    #define  mes(a, b)  memset(a, b, sizeof a)
    #define  fi         first
    #define  se         second
    #define  pb         push_back
    #define  pii        pair<int, int>
    
    typedef unsigned long long int ull;
    typedef long long int ll;
    const int    maxn = 1e6 + 10;
    const int    maxm = 1e5 + 10;
    const ll     mod  = 1e9 + 7;
    const ll     INF  = 1e18 + 100;
    const int    inf  = 0x3f3f3f3f;
    const double pi   = acos(-1.0);
    const double eps  = 1e-8;
    using namespace std;
    
    ll n, m;
    int cas, tol, T;
    
    int pri[maxn], phi[maxn];
    bool ispri[maxn];
    ll f[maxn], g[maxn], inv[maxn];
    
    void handle() {
        int mx = 1e6;
        mes(ispri, 1);
        tol = 0;
        phi[1] = 1;
        for(int i=2; i<=mx; i++) {
            if(ispri[i]) {
                pri[++tol] = i;
                phi[i] = i-1;
            }
            for(int j=1; j<=tol&&i*pri[j]<=mx; j++) {
                ispri[i*pri[j]] = 0;
                if(i%pri[j] == 0) {
                    phi[i*pri[j]] = phi[i]*pri[j];
                    break;
                } else {
                    phi[i*pri[j]] = phi[i]*(pri[j]-1);
                }
            }
        }
    }
    
    int main() {
        // freopen("in", "r", stdin);
        handle();
        inv[1] = 1;
        scanf("%d", &T);
        while(T--) {
            ll p;
            scanf("%lld%lld%lld", &n, &m, &p);
            ll x = min(n, m);
            for(int i=2; i<=x; i++) inv[i] = (p-p/i)*inv[p%i]%p;
            for(int i=1; i<=x; i++) f[i] = (n/i)*(m/i)%p;
            for(int i=x; i>=1; i--) {
                g[i] = f[i];
                for(int j=2; i*j<=x; j++) {
                    g[i] -= g[i*j];
                    if(g[i]<0)  g[i]+=p;
                }
            }
            ll ans = 0;
            for(int i=1; i<=x; i++) {
                ans += 1ll*g[i]*i%p * inv[phi[i]]%p;
                ans %= p;
            }
            printf("%lld
    ", ans);
        }
        return 0;
    }
    
  • 相关阅读:
    处理了一个“服务器能ping得通,但telnet连接失败”导致数据库登录不了的问题
    解决了一个oracle登录缓慢的问题
    今天解决了一个mysql远程登录和本机ip登录都失败的问题
    c++笔记
    c语言笔记
    常见并发与系统设计
    linux网络IO笔记
    linux文件IO全景解析
    linux网络协议笔记
    长大后才懂的蜡笔小新 ​​​​
  • 原文地址:https://www.cnblogs.com/Jiaaaaaaaqi/p/11423331.html
Copyright © 2011-2022 走看看