【BZOJ】2242: [SDOI2011]计算器

http://www.lydsy.com/JudgeOnline/problem.php?id=2242

题意:(前两个问略...)第三个问是,求$a^x \equiv b \pmod{p}$,它们范围都是$10^9$哒= =

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

typedef long long ll;
int mpow(ll a, ll b, ll p) {
	ll r=1; a%=p;
	while(b) { if(b&1) r=((ll)r*a)%p; a=((ll)a*a)%p; b>>=1; }
	return r;
}
void gcd(ll a, ll b, ll &d, ll &x, ll &y) {
	if(!b) { d=a; x=1; y=0; return; }
	gcd(b, a%b, d, y, x); y-=a/b*x;
}
void ni(ll a, ll b, ll p) {
	ll d, x, y, t;
	gcd(a, p, d, x, y); if(b%d) { puts("Orz, I cannot find x!"); return; }
	t=p/d;
	while(x<0) x+=t;
	while(x>=t) x-=t;
	printf("%lld\n", (x*b)%p);
}
map<int, int> s;
void bsgs(ll y, ll z, ll p) {
	y%=p; z%=p;
	if(z==1) { puts("0"); return; }
	if(!y && !z) { puts("1"); return; }
	if(!y) { puts("Orz, I cannot find x!"); return; }
	s.clear();
	int m=sqrt(p+0.5), t=1, w=y;
	for(int i=0; i<m; ++i) s[((ll)z*t)%p]=i, t=((ll)t*w)%p;
	w=mpow(y, m, p); y=1; t=(p-1)/m+1; bool flag=1;
	for(int i=0; i<=t; ++i) if(s.count(y)) { printf("%lld\n", (ll)m*i-s[y]); flag=0; break; } else y=((ll)y*w)%p;
	if(flag) puts("Orz, I cannot find x!");
}
int main() {
	int z, y, p, c, T;
	scanf("%d%d", &T, &c);
	while(T--) {
		scanf("%d%d%d", &y, &z, &p);
		if(c==1) printf("%d\n", mpow(y, z, p));
		else if(c==2) ni(y, z, p);
		else bsgs(y, z, p);
	}
	return 0;
}

  



bsgs裸题....其实就是一种分块思想..(为啥有那么牛的名字呢= =其实是我不想加分类了= =)即小块暴力然后大块就解决的思想,相信你们都能秒懂= =

首先我们随便选一个$m$,使得$x=km-t, <=0<t$,(这虽然有点区别于取余,但是这是为了方便= =)

然后推得

$$a^{km} \equiv ba^t \pmod{p}$$

然后就是右边暴力预处理,左边枚举$k$...由于枚举$k$复杂度是$O(n/m)$,显然取$m=\sqrt{n}$最优= =...由于懒,开个set记录右边= =于是总复杂度是$O(\sqrt{n}log(\sqrt{n}))$

时间: 2024-10-08 20:50:43

【BZOJ】2242: [SDOI2011]计算器的相关文章

BZOJ 2242: [SDOI2011]计算器( 快速幂 + 扩展欧几里德 + BSGS )

没什么好说的... --------------------------------------------------------------------- #include<cstdio> #include<cmath> #include<map> using namespace std; typedef long long ll; int MOD; void gcd(int a, int b, int& d, int& x, int& y)

bzoj 2242 [SDOI2011]计算器(数论知识)

Description 你被要求设计一个计算器完成以下三项任务: 1.给定y,z,p,计算Y^Z Mod P 的值: 2.给定y,z,p,计算满足xy≡ Z ( mod P )的最小非负整数: 3.给定y,z,p,计算满足Y^x ≡ Z ( mod P)的最小非负整数. Input 输入包含多组数据. 第一行包含两个正整数T,K分别表示数据组数和询问类型(对于一个测试点内的所有数据,询问类型相同). 以下行每行包含三个正整数y,z,p,描述一个询问. Output 对于每个询问,输出一行答案.对

BZOJ 2242 [SDOI2011]计算器(快速幂+Exgcd+BSGS)

[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2242 [题目大意] 给出T和K 对于K=1,计算 Y^Z Mod P 的值 对于K=2,计算满足 xy≡ Z ( mod P ) 的最小非负整数 对于K=3,计算满足 Y^x ≡ Z ( mod P) 的最小非负整数 [题解] K=1情况快速幂即可 K=2情况用exgcd求解 K=3用BSGS求解 [代码] #include <cstdio> #include <cmath&

[原博客] BZOJ 2242 [SDOI2011] 计算器

题目链接 noip级数论模版题了吧.让求三个东西: 给定y,z,p,计算Y^Z Mod P 的值. 给定y,z,p,计算满足xy≡ Z ( mod P )的最小非负整数. 给定y,z,p,计算满足Y^x ≡ Z ( mod P)的最小非负整数. 其中P均为素数.来分着处理. 1 y^z%p 快速幂.推荐一种又快又好写的写法. 1 LL power_mod(LL a,LL b,LL p){ //get a^b%p 2 LL ret=1; 3 while(b){ 4 if(b&1) ret = re

bzoj 2242: [SDOI2011]计算器 &amp; BSGS算法笔记

这题的主要难点在于第三问该如何解决 于是就要知道BSGS是怎样的一种方法了 首先BSGS是meet in the middle的一种(戳下面看) http://m.blog.csdn.net/blog/zentropy/11200099 看完链接后再看以下内容 --------------------------------------------------------------------------------------------------------------------- 对

BZOJ 2242 SDOI2011 计算器 快速幂+扩展欧几里得+BSGS

题目大意:--简洁明了自己看 第一问快速幂 第二问扩展欧几里得 第三问BSGS 顺便一开始没看到p是质数0.0 去弄了EXBSGS的模板0.0 懒得改了 #include<cmath> #include<cstdio> #include<cstring> #include<iostream> #include<algorithm> #define M 1001001 using namespace std; typedef long long l

BZOJ 2242 [SDOI2011]计算器 BSGS+快速幂+EXGCD

题意:链接 方法: BSGS+快速幂+EXGCD 解析: BSGS- 题解同上.. 代码: #include <cmath> #include <cstdio> #include <cstring> #include <iostream> #include <algorithm> #define MOD 140345 using namespace std; typedef long long ll; ll t,k,ans; ll y,z,p;

【快速幂】【扩展欧几里德】【BSGS】【SDOI 2011】【bzoj 2242】计算器

2242: [SDOI2011]计算器 Time Limit: 10 Sec Memory Limit: 512 MB Submit: 2077 Solved: 812 Description 你被要求设计一个计算器完成以下三项任务: 1.给定y,z,p,计算Y^Z Mod P 的值: 2.给定y,z,p,计算满足xy≡ Z ( mod P )的最小非负整数: 3.给定y,z,p,计算满足Y^x ≡ Z ( mod P)的最小非负整数. Input 输入包含多组数据. 第一行包含两个正整数T,K

BZOJ:2242: [SDOI2011]计算器

题解:BSGS 问题:map空间 BSGS判无解 a%p!=0 0与最小非负整数 有区别 函数传参类型转换int->long long long long ->int; 费马小定理充分必要 性? #include<iostream> #include<cstdio> #include<cstring> #include<map> #include<cmath> using namespace std; typedef long lon