zoukankan      html  css  js  c++  java
  • Tenka1-2017 F ModularPowerEquation!!【数论、扩欧】

    传送门

    给定自然数 (a) 和正整数 (m),求 (k<2cdot 10^{18}) 使得 (a^kequiv kpmod m)(T) 组数据。

    (Tle 100,a,mle 10^9)


    众所周知,(a^kmod m) 在不循环部分之后进入长度为 (varphi(m)) 的约数的循环。

    于是我们解方程均只考虑 (ge mvarphi(m)) 的解,就不用考虑开头部分了。

    考虑构造一个 (y),使得存在 (x),其中 (xequiv a^ypmod m)(xequiv ypmod{varphi(m)}),可以得到 (a^xequiv a^yequiv xpmod m)

    需要 (a^yequiv ypmod{gcd(m,varphi(m))}),这是一个递归的子问题。用扩欧可以从 (y) 算出 (x)

    复杂度瓶颈在算 (varphi),先筛出 (m) 的质因子再求,(O(frac{Tsqrt m}{log m}))

    #include<bits/stdc++.h>
    using namespace std;
    typedef long long LL;
    const int N = 31625, M = 3403;
    template<typename T>
    void read(T &x){
    	int ch = getchar(); x = 0; bool f = false;
    	for(;ch < '0' || ch > '9';ch = getchar()) f |= ch == '-';
    	for(;ch >= '0' && ch <= '9';ch = getchar()) x = x * 10 + ch - '0';
    	if(f) x = -x;
    } int ksm(int a, LL b, int mod){
    	int res = 1;
    	for(;b;b >>= 1, a = (LL)a * a % mod)
    		if(b & 1) res = (LL)res * a % mod;
    	return res;
    }
    int T, a, m, pri[M], tot, pr[15], cnt;
    bool notp[N];
    void init(int m){
    	notp[0] = notp[1] = true;
    	for(int i = 2;i <= m;++ i){
    		if(!notp[i]) pri[tot++] = i;
    		for(int j = 0;j < tot && i * pri[j] <= m;++ j){
    			notp[i * pri[j]] = true;
    			if(!(i % pri[j])) break;
    		}
    	}
    } void fact(int x){ cnt = 0;
    	for(int i = 0;i < tot && pri[i] * pri[i] <= x;++ i) if(!(x % pri[i])){
    		x /= pri[i]; pr[cnt++] = pri[i]; while(!(x % pri[i])) x /= pri[i];
    	} if(x > 1) pr[cnt++] = x;
    } int phi(int x){
    	int ans = x;
    	for(int i = 0;i < cnt && pr[i] * pr[i] <= x;++ i) if(!(x % pr[i])){
    		ans = ans / pr[i] * (pr[i] - 1); x /= pr[i];
    		while(!(x % pr[i])) x /= pr[i];
    	} if(x > 1) ans = ans / x * (x - 1);
    	return ans;
    } int exgcd(int a, int b, int &x, int &y){
    	if(!b){x = 1; y = 0; return a;}
    	int d = exgcd(b, a % b, y, x);
    	y -= a / b * x; return d;
    } LL calc(int a, int m){
    	if(m == 1) return 1;
    	int n = phi(m), x, y, d = exgcd(n, m, x, y);
    	LL u = calc(a, d), v = ksm(a, u, m);
    	LL ret = x * (v - u) / d % m, mod = (LL)m * n;
    	return (n * ret + u + mod) % mod + mod;
    }
    int main(){
    	read(T); init(N-1);
    	while(T --){
    		read(a); read(m); a %= m; fact(m);
    		printf("%lld
    ", calc(a, m));
    	}
    }
    
  • 相关阅读:
    协程初探
    属性传值
    分析代理模式
    上下文菜单与TrackPopupMenu
    走进小作坊(十六)----口碑营销
    线程池QueueUserWorkItem
    exosip
    Spark Core源代码分析: Spark任务运行模型
    微软2014校园招聘笔试试题
    怎样配置Tomcat环境变量
  • 原文地址:https://www.cnblogs.com/AThousandMoons/p/14607769.html
Copyright © 2011-2022 走看看