[CodeForces - 906D] Power Tower——扩展欧拉定理

题意

给你 $n$ 个 $w_i$ 和一个数 $p$,$q$个询问,每次询问一个区间 $[l,r] $,求 $w_l ^{w_{l+1}^{{\vdots}^{w_r}}} \ \% p$

分析

由扩展欧拉定理:

$$a^b\equiv \begin{cases} a^{b\%\phi(p)}~~~~~~~~~~~gcd(a,p)=1\\ a^b~~~~~~~~~~~~~~~~~~gcd(a,p)\neq1,b<\phi(p)\\ a^{b\%\phi(p)+\phi(p)}~~~~gcd(a,p)\neq1,b\geq\phi(p) \end{cases}~~~~~~~(mod~p)$$

与BZOJ 3384类似,但是在BZOJ 3384中,次方是无限的,所以说指数一定大于 $\varphi(p)$,但是这道题中指数不一定大于 $\varphi(p)$,需要重写 Mod。

phi需要记忆话,不然会超时。

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

typedef long long ll;
const int maxn = 1e5 + 10;
ll n, p, a[maxn];
unordered_map<int, int>phi;

ll Mod(ll x, ll mod)
{
    return x < mod ? x : x % mod + mod;
}

ll euler_phi(ll n)
{
    ll m = (ll)sqrt(n + 0.5);
    ll ans = n;
    for (ll i = 2; i <= m; i++)
    {
        if (n % i == 0)
        {
            ans = ans / i * (i - 1);
            while (n % i == 0)  n /= i;        //除尽
        }
    }
    if (n > 1)  ans = ans / n * (n - 1);    //剩下的不为1,也是素数
    return ans;
}

ll get_phi(ll x)
{
    if(phi[x])  return phi[x];
    return phi[x] = euler_phi(x);
}

ll qpow(ll a, ll b, ll p)
{
    ll ret = 1;
    while(b)
    {
        if(b&1) ret = Mod(ret * a, p);
        a = Mod(a * a ,p);
        b >>= 1;
    }
    return ret;
}

ll cal(ll l, ll r, ll p)   //a^a^a..^a共b次 
{
   //printf("%lld %lld\n", t, p);
    //if(t == 1)  return Mod(a, p);
    if(l == r)  return Mod(a[l], p);
    if(p == 1)  return Mod(a[l], p);
    ll phip = get_phi(p);
    return qpow(a[l], cal(l+1, r, phip), p);  //第一类和第三类
}

int main()
{
    scanf("%I64d%I64d", &n, &p);
    for(int i = 1;i <= n;i++)  scanf("%I64d", &a[i]);
    int q;
    scanf("%d", &q);
    while(q--)
    {
        ll l, r;
        scanf("%I64d%I64d", &l, &r);
        printf("%I64d\n", cal(l, r, p) % p);  //这个取模不能少
    }
    return 0;
}

 

 

参考链接:

1. https://blog.csdn.net/Charlie_jilei/article/details/79252689

2. https://blog.csdn.net/qq_35914587/article/details/79883547

posted @ 2019-09-02 13:00  Rogn  阅读(388)  评论(0编辑  收藏  举报