[SDOI2011][洛谷P2485]计算器(BSGS模板)

题面

https://www.luogu.com.cn/problem/P2485

题解

对于T=1,快速幂即可。
对于T=2,exgcd即可。
重点是T=3,需要使用BSGS算法。

BSGS算法

用来快速求解形如\({a^x}{\equiv}r(mod p)\)的指数同余方程的算法,适用范围是\(\gcd(a,p)=1\)
由欧拉-费马定理可得\(a^{\phi(p)}{\equiv}1(mod p)\)。下设\(0<=x<{\phi(p)}\)
设置“间隔”\(T\),那么x一定可以表示成\(iT-j\)(其中\(1<=j<=T\))的形式。下有

\[a^{iT-j}{\equiv}r{\mod p} \]

\[a^{iT}{\equiv}r*a^j{\mod p} \]

故只要某一个\(a^{iT}(1<=i<={\lceil{{\phi(p)-1}\over{T}}\rceil})\)以及某一个\(r*a^j(1<=j<=T)\)相等,就得到了一个解\(x=iT-j\)

实现时只需要将所有的\(r*a^j\)预处理好放入Hash表内,在计算\(a^{iT}\)的时候同步查询就可以了。

计算\(r*a^j\)的时间复杂度为\(O(T)\),计算\(a^{iT}\)的时间复杂度是\(O({{\phi(p)}\over{T}})\),取\(T=\sqrt{\phi(p)}\)即可保证总时间复杂度是\(O(\sqrt{\phi(p)})\)

回本题

给出\(y\),\(z\),\(p\),计算满足\(y^x{\equiv}z{\mod{p}}\)的最小非负整数 \(x\)

保证\(p\)是质数。

首先考虑\(p|y\)的情况,即\({\gcd}(y,p){\not=}1\)的情况,此时不能适用BSGS算法

  • 1、\(z=0\)\(x=1\)即可
  • 2、\(z=1\)\(x=0\)即可
  • 3、others:无解

其他情况可以适用BSGS解决。

代码

#include<bits/stdc++.h>

using namespace std;

#define ll long long
#define rg register

inline ll read(){
	ll s = 0,ww = 1;
	char ch = getchar();
	while(ch < '0' || ch > '9'){if(ch == '-')ww = -1;ch = getchar();}
	while('0' <= ch && ch <= '9'){s = 10 * s + ch - '0';ch = getchar();}
	return s * ww;
}

inline void write(ll x){
	if(x < 0)putchar('-'),x = -x;
	if(x > 9)write(x / 10);
	putchar('0' + x % 10);
}

ll mod,g;

inline ll check(ll x){
	return (x % mod + mod) % mod;
}

namespace ModCalc{
	inline void Inc(ll &x,ll y,ll mod){
		x += y;if(x >= mod)x -= mod;
	}
	
	inline void Dec(ll &x,ll y,ll mod){
		x -= y;if(x < 0)x += mod;
	}
	
	inline ll Add(ll x,ll y,ll mod){
		Inc(x,y,mod);return x;
	}
	
	inline ll Sub(ll x,ll y,ll mod){
		Dec(x,y,mod);return x;
	}
}
using namespace ModCalc;

inline ll power(ll a,ll n){
	ll x = a,s = 1;
	while(n){
		if(n & 1)s = s * x % mod;
		x = x * x % mod;
		n >>= 1;
	}
	return s;
}

inline ll exgcd(ll a,ll &x,ll b,ll &y){
	if(!b){
		x = 1,y = 0;
		return a;
	}
	ll g = exgcd(b,y,a%b,x);
	y -= a / b * x;
	return g; 
}

map<ll,ll>M;

int main(){
	ll T = read(),opt = read();
	if(opt == 1){
		while(T--){
			ll a = read(),n = read();
			mod = read();a %= mod;
			write(power(a,n));putchar('\n');
		}
	}
	else if(opt == 2){
		while(T--){
			ll a = read(),r = read();
			mod = read();
			ll x,y;
			ll g = exgcd(a,x,mod,y);
			if(r % g != 0){
				puts("Orz, I cannot find x!");
				continue;
			}
			a /= g,r /= g,mod /= g;
			x = check(x * r);
			write(x);putchar('\n');
		}
	}
	else{
		while(T--){
			ll a = read(),r = read();
			mod = read();a %= mod,r %= mod;
			if(!a){
				if(!r)puts("1");
				else if(r == 1)puts("0");
				else puts("Orz, I cannot find x!");
				continue;
			}
			ll T = (ll)round(sqrt(mod));
			ll a_T = power(a,T);
			M.clear();
			for(rg ll i = 1,cur = r * a % mod;i <= T;i++,cur = cur * a % mod)M[cur] = i;
			ll flag = 0;
			for(rg ll i = 1,cur = a_T;i * T - T <= mod - 2;i++,cur = cur * a_T % mod)if(M.count(cur)){
				write(Sub(i*T%(mod-1),M[cur],mod-1)),putchar('\n');
				flag = 1;
				break;
			}
			if(!flag)puts("Orz, I cannot find x!");
		}
	}
	return 0;
}

posted @ 2020-02-03 11:53  coder66  阅读(191)  评论(0编辑  收藏  举报