\(\\\)

BSGS


用于求解关于 \(x\) 的方程:

\[a^x\equiv b\pmod p\ ,\ (p,a)=1
\]

一般求解的是模意义下的指数,也就是最小非负整数解。

\(\\\)

算法思想


本质是双向搜索,或阈值优化的思想。

首先设"步幅" 为 \(m=\lceil{ \sqrt p}\rceil\) ,然后将方程写作

\[a^{i\times m-j}\equiv b\pmod p
\]

其中 \(i\) 就是所谓"大步", \(j\) 就是所谓"小步",我们要把他们组合在一起。

直接搜索两个数不如折半搜索一个数,然后再组合。

于是我们可以将分母上的 \(a^j\) 移项,得到

\[a^{i\times m}\equiv b\times a^j\pmod p
\]

然后就成了比较标准的双向搜索形式。

先把右一半的答案记下来,然后拿左一半搜到的每一个数去查询是否出现过就好了。

\(\\\)

代码实现


对于每一个 \(j\in [0,m-1]\) ,将 \(b\times a^j\ \%\ p\) 的答案放到哈希表里。

然后对于每一个 \(i\in[1,m](\) 此范围依据定义而来,尤其注意!\()\),去哈希表里查是否有 \(a^{im}\ \%\ p\) 的值。

还有两个小优化:

  • 注意到求出为同一个值的 \(j\) ,因为在答案里系数为 \(-1\) ,所以对于求出最小解 \(j\) 肯定是越大越优秀。

    因此再哈希表里插入相同的值时,可以直接取 \(max\), 如果是按序插入直接覆盖即可。

    这里也延申出了一种做法,直接用 \(map\) 存储结果,将结果映射到 \(j\) ,按序插入直接覆盖,复杂度多个\(log\) 。

  • 运算过程中只需一次快速幂。

    一开始每一次都是乘上 \(a\) ,所以一遍循环一遍乘即可,第二步同理,只需题前计算出 \(a^m\) 的值。

    这一优化在需要快速乘的时候效果很好。

我们以 [TJOI2007]可爱的质数 一题为例提供一份模板。

  1. #include<map>
  2. #include<cmath>
  3. #include<cstdio>
  4. #include<cctype>
  5. #include<cstring>
  6. #include<cstdlib>
  7. #include<iostream>
  8. #include<algorithm>
  9. #define R register
  10. using namespace std;
  11. typedef long long ll;
  12. map<ll,ll> s;
  13. inline ll qpow(ll x,ll t,ll p){
  14. ll res=1;
  15. while(t){
  16. if(t&1) (res*=x)%=p;
  17. (x*=x)%=p; t>>=1;
  18. }
  19. return res;
  20. }
  21. inline ll BSGS(ll a,ll b,ll p){
  22. b%=p;
  23. ll m=ceil(sqrt(p));
  24. for(R ll i=0;i<m;++i,(b*=a)%=p) s[b]=i;
  25. a=qpow(a,m,p);
  26. for(R ll i=1,tmp=a;i<=m;++i,(tmp*=a)%=p)
  27. if(s.find(tmp)!=s.end()){
  28. if(i*m<s[tmp]) continue;
  29. return i*m-s[tmp];
  30. }
  31. return -1;
  32. }
  33. int main(){
  34. ll a,b,p;
  35. scanf("%lld%lld%lld",&p,&a,&b);
  36. ll x=BSGS(a,b,p);
  37. if(x>=0) printf("%lld\n",x);
  38. else puts("no solution");
  39. return 0;
  40. }

大步小步法(BSGS) 学习笔记的更多相关文章

  1. 2022-07-10 第五小组 pan小堂 css学习笔记

    css学习笔记 什么是 CSS? CSS 指的是层叠样式表* (Cascading Style Sheets) CSS 描述了如何在屏幕.纸张或其他媒体上显示 HTML 元素 CSS 节省了大量工作. ...

  2. BSGS 扩展大步小步法解决离散对数问题 (BZOJ 3239: Discrete Logging// 2480: Spoj3105 Mod)

    我先转为敬? orz% miskcoo 贴板子 BZOJ 3239: Discrete Logging//2480: Spoj3105 Mod(两道题输入不同,我这里只贴了3239的代码) CODE ...

  3. 微信小程序视频学习笔记

    [清华大学]学做小程序 https://www.bilibili.com/video/av21987398 2.2创建项目和文件结构 小程序包含一个描述整体程序的app和多个描述各自页面的page 配 ...

  4. BSGS学习笔记

    用于求\(A^{x} \equiv B \pmod{C}\) 高次方程的最小正整数解x,其中C为素数 引理1:$a^{i\mod\varphi(p) } \equiv a^{i} $ (mod p) ...

  5. 第1-5章 慕课网微信小程序开发学习笔记

    第1章 前言:不同的时代,不同的Web --微信小程序商城构建全栈应用 http://note.youdao.com/noteshare?id=a0e9b058853dbccf886c1a890594 ...

  6. 小甲鱼Python学习笔记

    一 isdigit()True: Unicode数字,byte数字(单字节),全角数字(双字节),罗马数字False: 汉字数字Error: 无 isdecimal()True: Unicode数字, ...

  7. BSGS 学习笔记

    问题:求$a^x\equiv b\ (mod\ p)$的最小正整数解. 这时候就要用到BSGS(拔山盖世)算法.直接进入正题: 设$x=im-n$, 则原式等于$a^{im-n}\equiv b\ ( ...

  8. 第6章 AOP与全局异常处理6.1-6.4 慕课网微信小程序开发学习笔记

    第6章 AOP与全局异常处理 https://coding.imooc.com/learn/list/97.html 目录: 第6章 AOP与全局异常处理6-1 正确理解异常处理流程 13:236-2 ...

  9. 第7章 数据库访问与ORM 慕课网微信小程序开发学习笔记

    第7章 数据库访问与ORM https://coding.imooc.com/learn/list/97.html 目录: 7-1 数据库操作三种方式之原生SQL 19:09 7-2 从一个错误了解E ...

随机推荐

  1. QtQuick桌面应用开发指导 1)关于教程 2)原型和设计 3)实现UI和功能_A

    Release1.0 http://qt-project.org/wiki/developer-guides Qt Quick Application Developer Guide for Desk ...

  2. IDEA失效的解决办法

    1.根据下图进行操作即可解决

  3. [IT学习]华为全连接大会2017

    1.5分钟.3分钟.1分钟倒计时. 2.20万盏纽约街头的油灯接入电网,类比未来的公司IT系统会接入云? 3.1943年,全球只要5台计算机.不会的,但是会有5多云? 4.与航空业的联盟类比,云计算的 ...

  4. asp.net 实现搜索站内搜索功能

    首先有index和search 两个页面 index页面中有textbox1和button1两个控件 双击button1控件添加代码: protected void Button1_Click(obj ...

  5. 城域网IPv6过渡技术—NAT64+DNS64 Test for IPv6 DNS64/NAT64 Compatibility Regularly

    城域网IPv6过渡技术—NAT64+DNS64 - 51CTO.COM http://network.51cto.com/art/201311/419623.htm Supporting IPv6 D ...

  6. shell 读写远程数据库

    http://www.cnblogs.com/wangkangluo1/archive/2012/04/27/2472898.html 利用Shell脚本实现远程MySQL自动查询 目的:对定时任务对 ...

  7. 滑动窗体的最大值(STL的应用+剑指offer)

    滑动窗体的最大值 參与人数:767时间限制:1秒空间限制:32768K 通过比例:21.61% 最佳记录:0 ms|8552K(来自 ) 题目描写叙述 给定一个数组和滑动窗体的大小.找出全部滑动窗体里 ...

  8. C语言中string char int类型转换

    C语言中string -- ::) 转载 ▼ 标签: 操作符 int char c语言 类型转换 分类: C/Cpp ,char型数字转换为int型 "; printf(]-');//输出结 ...

  9. Android连接wifi,调用系统API【转】

    本文转载自:http://blog.csdn.net/aaa1050070637/article/details/54136472 直接上代码,简单粗暴,一看就懂 import android.con ...

  10. linux进程编程入门

    1.进程的创建与操作 任务描述: 在父进程中创建一个全局变量,一个局部变量,并赋予初始值,用fork函数创建子进程.在子进程中对父进程的变量进行自加操作,并且输出变量值,然后父进程睡眠一段时间 各进程 ...