2019牛客多校第三场D BigInteger——基础数论

题意:

用  $A(n)$ 表示第 $n$ 个只由1组成分整数,现给定一个素数 $p$,求满足 $1 \leq i\leq n, 1 \leq j \leq m, A(i^j) \equiv 0(mod \ p)$ 的 $(i, j)$ 对数。

分析:

$11...11 = \frac{10^n-1}{9} \equiv 0(mod \ p)$ 等价于 $10^n \equiv 1(mod \ 9p)$,当 $p \neq 2,5$ 时,有 $gcd(10, 9p)=1$,因此 $10^{\phi(9p)} \equiv 1(mod \ 9p)$。我们需要找到满足等式最小的数 $d$,也是循环节,显然 $d \ | \ \phi (9p)$,直接枚举 $\phi(9p)$ 的约数验证即可。

找到循环节 $d$ 后,我们需要知道有多少对 $(i, j)$ 满足 $d \ | \ i^j$.

对 $d$ 做质因数分解, $d = p_1^{k_1}p_2^{k_2}...p_l^{k_l}$,考虑 $j$ 固定的时候,$i$ 需要满足什么条件?$i$ 必须是 $g = p_1^{\left \lceil \frac{K_1}{j} \right \rceil} p_2^{\left \lceil \frac{K_2}{j} \right \rceil} ... \ p_l^{\left \lceil \frac{K_l}{j} \right \rceil}$ 的倍数,因此共有 $\frac{n}{g}$ 个合法的 $i$。

由于 $k_i \leq 30$,所以 $j$ 增加到30以上和 $j=30$ 的结果是一样的,枚举 $j$ 从1到30,分别计算 $g$ 即可。

当 $p=2,5$ 的时候,显然答案为0.

$\phi(9p)$ 也不必用欧拉函数计算,当 $p \neq 3$ 时,3与p互素,根据欧拉函数的积性,$\phi (9p) = \phi (9)\phi (p) = 6(p-1)$.

由于快速幂会爆long long,需要用__int128(血的教训啊,wa了好多发,枯了)

代码:

#include<bits/stdc++.h>
using namespace std;

typedef long long ll;

ll p, m ,n;
map<ll, ll>ma;

__int128  qpow(__int128  a, __int128  b, __int128  p)
{
    __int128  res = 1;
    while(b)
    {
        if(b & 1)  res = res * a % p;
        a = (a * a) % p;  //a*a会爆long long
        b >>= 1;
    }
    return res;
}

ll  qpow2(ll  a, ll  b)

{
    ll  res = 1;
    while(b)
    {
        if(b & 1)  res = res * a;
        a = a * a;
        b >>= 1;
    }
    return res;
}

//约数枚举O(√n)
ll divisor(ll n, ll p)
{
    vector<ll>res;
    for (ll i = 1; i * i <= n; i++)
    {
        if (n % i == 0)
        {
            //printf("i:%lld\n", i);
            if(qpow(10, i, 9*p) % (9*p) == 1)  return i;
            if(i != n / i)  res.push_back(n / i);
        }
    }
    for(ll i = res.size()-1;i >= 0;i--)
    {
        // printf("i:%lld\n", res[i]);
         if(qpow(10, res[i], 9*p) % (9*p) == 1)  return res[i];
    }
    return 0;
}

//整数分解O(√n)
void prime_factor(ll n)
{
    for (int i = 2; i * i<= n; i++)
    {
        while (n % i == 0)
        {
            ++ma[i];
            n /= i;
        }
    }
    if (n != 1)  ma[n] = 1;        //最多只有一个素因数大于√n
}


//j固定的情况下的对数
ll  OneJ(int j)
{
    ll res = 1;
    for(auto it = ma.begin();it != ma.end();it++)
    {
        res *= qpow2((*it).first, (ll)ceil((*it).second * 1.0 / j));
    }
    return n / res;
}


int  main()
{
    int T;
    scanf("%d", &T);
    while(T--)
    {
        scanf("%lld%lld%lld", &p, &n, &m);
        //int fai = euler_phi(9*p);
        ll k;
        if(p == 2 || p == 5)
        {
            printf("0\n");
            continue;
        }

        if(p == 3)
        {
            k =  divisor(18, p);
        }
        else  k = divisor(6*(p-1), p);  //printf("k:%lld\n", k);

        if(k == 0)  printf("0\n");
        else
        {   ma.clear();
            ll res = 0;
            prime_factor(k);  //printf("k:%lld\n", k);

            if(m < 30)
            {
                 for(int i = 1;i <= m;i++)  res += OneJ(i);
            }
            else
            {
                for(int i = 1;i <= 29;i++)  res += OneJ(i);
                int tmp = OneJ(30);
                res += tmp * (m - 29);
            }
            printf("%lld\n", res);
        }
    }
    return 0;
}

 

posted @ 2019-07-28 14:14  Rogn  阅读(346)  评论(0编辑  收藏  举报