[置顶] hdu2815 扩展Baby step,Giant step入门
题意:求满足a^x=b(mod n)的最小的整数x。
分析:很多地方写到n是素数的时候可以用Baby step,Giant step, 其实研究过Baby step,Giant step算法以后,你会发现 它能解决 “n与a互质”的情况,而并不是单纯的n是素数的情况。如果a与n不是互质的,那么我们需要处理一下原方程,让a与n互质,然后再用Baby step,Giant step解出x即可。
Baby step,Giant step算法思想:对于a与n互质,那么则有a^phi(n)=1(mod n), 对于n是素数phi(n) == n-1, 否则phi(n) < n-1, 所以x的取值只要在0----n-2之中取就可以了。
当n很小时,可以直接枚举,但当n很大时,肯定会超时,Baby step,Giant step就是用了一种O(sqrt(n)*log(n))的方法枚举了所有的0-----n-2。令m = sqrt(n);
我们可以预处理出a^0,a^1,.........a^m,都放入哈希表中, 然后 (a^m)^i+v(哈希表里的其中一个值)就一定是解,每次枚举i(0-----m-1),计算出v,判断v是否出现在哈希表中,如果有就是解。 对于m为什么取sqrt(n)是为了复杂度的平衡,这一点是跟分块算法很相似的。
对于a与n不互质的情况分析:令 t = gcd(a,n),那么a与n都约去t,当然b也要约去t(不能约去就无解),约去一个t以后方程就变为 aa*a^(x-1) = bb(mod nn), (其中 aa = a/t bb = b/t nn = n/t) , 这里nn还可能与a不互质,那么我们一直拿出一个新的a对(a, bb, nn)约去t,直到a与nnn....(nnn...表示约去若干次t以后的n)互质。以下用(用三个字母表示约去若干次后,如bbb) 则结果为aa^c*a^(x-c) = bbb(mod nnn), 我们让等式左右分别乘以aa^c关于nnn的逆元 变为a^(x-c) = w (mod nnn) , w =bbb *(aa^c)^(-1)。 a^x = w (mod n)可以用bbb *(aa^c)^(-1)Baby step,Giant step直接求出,如果有解那把未知数+c。
具体看代码中的cal函数。
注意:在以上过程中x有可能<c,所以我们必须每约去一个t就要特判一下当前情况aa 与 bb就说明当前c是解。
哈希表实现看题目时间要求,map太慢,自己手写hash是很快的。
map哈希
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <map>
using namespace std;
typedef long long ll;
void ex_gcd(ll a, ll b, ll &x, ll &y) {
if (!b) {
x = 1;
y = 0;
} else {
ex_gcd(b, a % b, y, x);
y -= a / b * x;
}
}
inline ll inv(ll a, ll n) {
ll x, y;
ex_gcd(a, n, x, y);
return (x + n) % n;
}
ll log_mod(ll a, ll b, ll n) {
ll m, e;
int i;
m = sqrt(n + 0.5);
map<ll, ll> f;
f[1] = 0;
e = 1;
for (i = 1; i < m; i++) {
e = e * a % n;
if (!f.count(e))
f[e] = i;
}
e = e * a % n;
e = inv(e, n);
for (i = 0; i < m; i++) {
if (f.count(b))
return i * m + f[b];
b = b * e % n;
}
return -1;
}
ll gcd(ll a, ll b) {
return b ? gcd(b, a % b) : a;
}
ll cal(ll a, ll b, ll n) {
ll t, c = 0, v = 1;
while ((t = gcd(a, n)) != 1) {
if (b % t)
return -1;
n /= t;
v = v * a / t % n;
b /= t;
c++;
if (b == v)
return c;
}
//printf("a = %I64d b = %I64d c = %I64d v = %I64d\n", a, b, c, v);
b *= inv(v, n);
b %= n;
ll ret = log_mod(a, b, n);
return ~ret ? ret + c : ret;
}
int a, b, n;
int main() {
while (~scanf("%d%d%d", &a, &n, &b)) {
if (b >= n) {
printf("Orz,I can’t find D!\n");
continue;
}
if (b == 0) {
printf("0\n");
continue;
}
ll ans = cal(a, b, n);
if (ans == -1)
printf("Orz,I can’t find D!\n");
else
printf("%I64d\n", ans);
}
return 0;
}
手写hash
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <map>
using namespace std;
typedef long long ll;
const int maxn = 100007;
struct hash {
int n;
struct Edge {
int p, v;
int next;
}edge[maxn*7];
int head[maxn+10], E;
void init(int n) {
this->n = n;
memset(head, -1, sizeof(int)*(n+1));
}
void add(int p, int v) {
int s = p%n;
edge[E].p = p;
edge[E].v = v;
edge[E].next = head[s];
head[s] = E++;
}
int get(int p) {
int s = p%n;
for(int i = head[s]; ~i; i = edge[i].next) {
if(edge[i].p == p) return edge[i].v;
}
return -1;
} }f;
void ex_gcd(ll a, ll b, ll &x, ll &y) {
if (!b) {
x = 1;
y = 0;
} else {
ex_gcd(b, a % b, y, x);
y -= a / b * x;
}
}
inline ll inv(ll a, ll n) {
ll x, y;
ex_gcd(a, n, x, y);
if(x < 0) x += n;
return x;
}
ll log_mod(ll a, ll b, ll n) {
ll m, e;
int i;
m = sqrt(n + 0.5);
f.init(10007);
f.add(1, 0);
e = 1;
for (i = 1; i < m; i++) {
e = e * a % n;
if (f.get(e) == -1) f.add(e, i);
}
e = e * a % n;
e = inv(e, n);
for (i = 0; i < m; i++) {
int t = f.get(b);
if (~t)
return i * m + t;
b = b * e % n;
}
return -1;
}
ll gcd(ll a, ll b) {
return b ? gcd(b, a % b) : a;
}
ll cal(ll a, ll b, ll n) { //扩展函数
ll t, c = 0, v = 1;
while ((t = gcd(a, n)) != 1) {
if (b % t)
return -1;
n /= t;
b /= t;
v = v * a / t % n;
c++;
if (b == v) return c;
}
b = b*inv(v, n)%n;
ll ret = log_mod(a, b, n);
return ~ret ? ret + c : ret;
}
int a, b, n;
int main() {
while (~scanf("%d%d%d", &a, &n, &b)) {
if (b >= n) {
printf("Orz,I can’t find D!\n");
continue;
}
if (b == 0) {
printf("0\n");
continue;
}
ll ans = cal(a, b, n);
if (ans == -1)
printf("Orz,I can’t find D!\n");
else
printf("%I64d\n", ans);
}
return 0;
}
[置顶] hdu2815 扩展Baby step,Giant step入门的更多相关文章
- HDU 2815 扩展baby step giant step 算法
题目大意就是求 a^x = b(mod c) 中的x 用一般的baby step giant step 算法会超时 这里参考的是http://hi.baidu.com/aekdycoin/item/2 ...
- 【学习笔记】Baby Step Giant Step算法及其扩展
1. 引入 Baby Step Giant Step算法(简称BSGS),用于求解形如\(a^x\equiv b\pmod p\)(\(a,b,p\in \mathbb{N}\))的同余方程,即著名的 ...
- 解高次同余方程 (A^x=B(mod C),0<=x<C)Baby Step Giant Step算法
先给出我所参考的两个链接: http://hi.baidu.com/aekdycoin/item/236937318413c680c2cf29d4 (AC神,数论帝 扩展Baby Step Gian ...
- HDU 2815 Mod Tree 离散对数 扩张Baby Step Giant Step算法
联系:http://acm.hdu.edu.cn/showproblem.php?pid=2815 意甲冠军: watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQ ...
- 『高次同余方程 Baby Step Giant Step算法』
高次同余方程 一般来说,高次同余方程分\(a^x \equiv b(mod\ p)\)和\(x^a \equiv b(mod\ p)\)两种,其中后者的难度较大,本片博客仅将介绍第一类方程的解决方法. ...
- POJ 3243 Clever Y (求解高次同余方程A^x=B(mod C) Baby Step Giant Step算法)
不理解Baby Step Giant Step算法,请戳: http://www.cnblogs.com/chenxiwenruo/p/3554885.html #include <iostre ...
- 数论之高次同余方程(Baby Step Giant Step + 拓展BSGS)
什么叫高次同余方程?说白了就是解决这样一个问题: A^x=B(mod C),求最小的x值. baby step giant step算法 题目条件:C是素数(事实上,A与C互质就可以.为什么?在BSG ...
- 【POJ2417】baby step giant step
最近在学习数论,然而发现之前学的baby step giant step又忘了,于是去翻了翻以前的代码,又复习了一下. 觉得总是忘记是因为没有彻底理解啊. 注意baby step giant step ...
- POJ 2417 Discrete Logging ( Baby step giant step )
Discrete Logging Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 3696 Accepted: 1727 ...
随机推荐
- 强烈推荐一款CSS导航菜单
强烈推荐一款CSS导航菜单,用到政府学校类网站上超级不错,有点类似站长网菜单的味道,只不过颜色不一样而已,这种菜单还不是真正意义上的“下拉”菜单,应该叫滑出菜单吧?反正比较不错,不多说了. <! ...
- 关于URL编码的问题
在进行WEB开发时,字符集编码常常困扰着我们.我们需要区分两种情况,一是URL编码,二是HTTP Body编码.这两种编码所处理的机制不同. URL编码和解码 客户端负责对URL编码,服务端负责解码. ...
- WIN7 Wireshark: There are no interfaces on which a capture can be done
有的时候我们在Windows7的环境下使用Wireshark的时候,比如点击[Interface List]的时候,出现错误. 错误内容如下: There are no interfaces on w ...
- java MongoDB driver error infos
DataTables warning: table id=dateTable - Ajax error. For more information about this error, please s ...
- TimeSpan类【转】
TimeSpan ts1 = new TimeSpan(DateTime.Now.Ticks); //获取当前时间的刻度数 //执行某操作 ............................ . ...
- huffman 编码
huffman压缩是一种压缩算法,其中经典的部分就是根据字符出现的频率建立huffman树,然后根据huffman树的构建结果标示每个字符.huffman编码也称为前缀编码,就是每个字符的表示形式不是 ...
- win7下安装Ubuntukylin-14.04双系统
工具准备: 下载ISO系统镜像,UltraISO,EasyBCD,分区助手,8G 优盘 U盘启动制作流程: 1,打开分区助手,从硬盘中分出空闲空间(60G)作为Ubuntu工作空间,文件系统设为Ext ...
- 启动监听报错:TNS-12537: TNS:connection closed TNS-12560: TNS:protocol adapter error TNS-00507: Connection closed Linux Error: 29: Illegal seek
启动监听程序报错: 说明:在rhel5.8上安装完成oracle11g数据库后,使用netca创建完监听,启动监听时报错.还未使用dbca创建实例. [oracle@rusky-oracle11g ~ ...
- jqery ajax读取json文件
json文件数据 [ {"name":"哈哈··","email":"邮箱01","gender": ...
- 图的最短路径问题————树上奶牛(tree.cpp)
和往常一样,继续从题目引入 树上奶牛 (tree.cpp) [题目描述] 农夫John的奶牛不是住在地上而是住在树上的QWQ. 奶牛之间需要串门,不过在串门之前他们会向John询问距离的大小.可是Jo ...