离散对数及其拓展

离散对数是在群Zp∗Z_{p}^{*}Zp∗​而言的,其中ppp是素数。即在在群Zp∗Z_{p}^{*}Zp∗​内,aaa是生成元,求关于xxx的方程ax=ba^x=bax=b的解,并将解记作x=logabx=log_{a}{b}x=loga​b,离散对数指的就是这个logablog_{a}{b}loga​b.由于群Zp∗Z_{p}^{*}Zp∗​的阶是p−1p-1p−1,且是循环群,因为生成元的阶是p−1p-1p−1,因而模p−1p-1p−1相等的指数可以看做一样的数,x=logabx=log_{a}{b}x=loga​b的值不妨定为Zp−1Z_{p-1}Zp−1​的元素,即模p−1p-1p−1的数,即0,1,2,3,…,p−20,1,2,3,\ldots,p-20,1,2,3,…,p−2.

以上说的是数学上对离散对数的定义,但是,搞算法竞赛人说的离散对数和这个有点差异,差异在于没有要求aaa是生成元。

数学上的离散对数的定义,解一定存在且解模p−1p-1p−1意义下唯一,而我们平常说的离散对数由于没有要求aaa是生成元,那么解可能不存在,解模p−1p-1p−1意义下不唯一,例如群Z7∗Z_7^*Z7∗​中222不是生成元(因为23=12^3=123=1),于是以222为底的离散对数,模666意义下解不唯一,模333意义下解才唯一;而且当b=3,5,6b=3,5,6b=3,5,6的时候没有解。

所以算法竞赛中说的离散对数一般是任意的aaa,若有解求最小非负整数解,否则返回无解。

如何求离散对数的值?由于答案在Zp−1Z_{p-1}Zp−1​内,最简单暴力的做法是枚举Zp−1Z_{p-1}Zp−1​中的每个数,共p−1p-1p−1个数,复杂度O(p)O(p)O(p).

更高效的做法------Shank的大步小步算法

注意到Zp∗Z_{p}^{*}Zp∗​是个群,所以会有很多很好的性质,例如每个元素都有逆元。

我们选取一个数s,然后任意指数xxx都可以表示成x=ks+r(0≤r&lt;s)x=ks+r(0 \leq r &lt; s)x=ks+r(0≤r<s).在群内,有------

ax=baks+r=b两边乘a−ksar=a−ksbar=(a−s)kb\begin{aligned}
a^x &amp;= b \\
a^{ks+r} &amp;= b \quad\quad\quad\quad\quad \text{两边乘}a^{-ks}\\
a^r&amp;=a^{-ks}b\\
a^r&amp;= \left(a^{-s}\right)^{k}b
\end{aligned}axaks+rarar​=b=b两边乘a−ks=a−ksb=(a−s)kb​

每一个指数xxx都可以和一个有序对(k,r)(k,r)(k,r)建立一一对应关系,求xxx就变成确定(k,r)(k,r)(k,r)的问题了。注意上面的式子最后一行,左边ar(0≤r&lt;k)a^r(0 \leq r &lt; k)ar(0≤r<k)最多有sss个取值,右边a−ka^{-k}a−k是固定的,而0≤k≤p−2s0 \leq k \leq \frac{p-2}{s}0≤k≤sp−2​,因此右边最多有t=1+⌈p−2s⌉t=1+\lceil \frac{p-2}{s}\rceilt=1+⌈sp−2​⌉个取值。

于是变成了右边ttt个数在左边sss个数中有没有相等的元素.

这便是Shank的大步小步算法(Shank’s Baby-Step-Giant-Step

Algorithm)。左边的rrr指数是每一加1,是小步;右边指数每次的变化是sss,是大步。

为了求最小的xxx,具体地我们要从小到大枚举kkk,计算出右边的值,查询左边sss个数中是否有相等的数。若有,符合条件最小的rrr与kkk即构成最小的解xxx;若无,则看下一个kkk;若所有kkk都看完了还没有,那么就说明无解。

为了加速查询,显然对左边的sss个数需要预先计算出来,并建立由值查询最小rrr的索引"数组",如果直接用值做下标的话,空间需要O(p)O(p)O(p),但是总共才sss个值,所以应该用map建立"数组"或者hash(这个写麻烦一点)。

计算左边和右边所有取值的总的复杂度是O(s+t)O(s+t)O(s+t).插入及查询的时间复杂度使用map是O((s+t)ln⁡r)O((s+t)\ln{r})O((s+t)lnr),使用hash是O(s+t)O(s+t)O(s+t)。空间复杂度是索引数组大小O(s)O(s)O(s)。而t=1+⌈p−2s⌉t=1+\lceil \frac{p-2}{s}\rceilt=1+⌈sp−2​⌉,取s=ps=\sqrt{p}s=p​附近的数,可以获得比较好的复杂度,如果使用map的话就是O(pln⁡p)O(\sqrt{p}\ln{p})O(p​lnp),如果是使用hash的话就是O(p)O(\sqrt{p})O(p​).

如果map会被卡就换成hash吧。

大步小步思路

略微思考便可以发现大步小步算法的精妙在于将原本需要的nnn次计算的问题变成了左右各O(n)O(\sqrt{n})O(n​)次的计算加O(n)O(\sqrt{n})O(n​)次插入和查询问题。**这可以说是一种牺牲空间换时间的一种思路。**当我们去掉a,ba,ba,b在这里代表的数含义的话,如果还可以有相同的变换,那么也可以如此降低复杂度。

另外,一点小优化,x=ks+rx=ks+rx=ks+r也可以换成x=ks−r(k≥1,1≤r≤s)x=ks-r \quad(k \geq 1,1 \leq r \leq s)x=ks−r(k≥1,1≤r≤s)的形式,然后将式子变换成(as)k=bar(a^s)^k=ba^r(as)k=bar.如此,就不用求逆元了。

会发现上面式子的变换只用到了群的性质,所以Zp∗Z_p^*Zp∗​换成Zn∗Z_n^*Zn∗​依旧可以如此做,只是阶不是由p−1p-1p−1变成了n−1n-1n−1,而是变成了ϕ(n)\phi(n)ϕ(n),事实上p−1p-1p−1只是ppp是质数时的ϕ(p)\phi(p)ϕ(p)。当然,指数算到ϕ(n)−1\phi(n)-1ϕ(n)−1即可,当然,算到n−2n-2n−2也无妨,只是多算了一点而已,不影响结果。

当然,这就要求aaa必须是Zn∗Z_n^*Zn∗​中的元素了,即(a,n)=1(a,n)=1(a,n)=1,即互质。

拓展离散对数的思考与推导

离散对数的拓展,其实是想解ax≡b(modn)a^x \equiv b \pmod{n}ax≡b(modn)的方程中最小的自然数解x,其中aaa是一般数,不保证与nnn互质。因此必须思考aaa与nnn不互质如何解的问题。

基本思路是,通过方程的等价变化,将暂时无法解决的问题化为可以解决的问题。

[如果只想看结论可以直接往下看下一小节的具体算法部分。]{.underline}

不妨考虑成qax≡b(modn)qa^x \equiv b \pmod{n}qax≡b(modn)形式的问题,首先将其视作关于axa^xax的线性同余方程,线性同余方程有解等价于(q,n)∣b(q,n) \mid b(q,n)∣b.显然关于axa^xax的线性同余方程有解是原方程有解的必要条件,此必要条件成立时,qax≡b(modn)qa^x \equiv b \pmod{n}qax≡b(modn)显然等价于qa(a,x)ax−1≡b(a,n)(modn(a,n))\frac{qa}{(a,x)}a^{x-1} \equiv \frac{b}{(a,n)} \pmod{\frac{n}{(a,n)}}(a,x)qa​ax−1≡(a,n)b​(mod(a,n)n​)。新的方程和原方程形式上相同,但是模数变小了。如此,只要aaa与模数不互质,就重复进行如此的变换,最终一定会终止于互质的情况(当然如果某一步等价变换的必要条件不满足则直接可以得出无解结论而提前终止了)。

假设最后终止于qcax−c≡bc(modnc)q_ca^{x-c} \equiv b_c \pmod{n_c}qc​ax−c≡bc​(modnc​),aaa与nnn已经互质,则大步小步算法解之即可。

当指数xxx在整个整数上域取值,这一系列的变换显然等价。

拓展离散对数的具体算法

解方程ax≡b(modn)a^x \equiv b \pmod{n}ax≡b(modn)的最小自然数解。

  1. 方程等价变换

    1. 将原方程通过等价变换,始终保持着qian−i≡bi(modni)q_{i}a^{n-i} \equiv b_i \pmod{n_i}qi​an−i≡bi​(modni​)的形式。

    2. 令q0=1,b0=b,n0=nq_0=1,b_0=b,n_0=nq0​=1,b0​=b,n0​=n。

    3. 迭代过程,qi+1=qia(a,ni),bi+1=bi(a,ni),ni+1=ni(a,ni)q_{i+1}=q_i\frac{a}{(a,n_i)},b_{i+1}=\frac{b_i}{(a,n_i)},n_{i+1}=\frac{n_i}{(a,n_i)}qi+1​=qi​(a,ni​)a​,bi+1​=(a,ni​)bi​​,ni+1​=(a,ni​)ni​​,迭代的条件是bi+1=bi(a,ni)b_{i+1}=\frac{b_i}{(a,n_i)}bi+1​=(a,ni​)bi​​是整数。

    4. 迭代终止条件:

      1. 不满足迭代条件返回无解

      2. (a,ni)=1(a,n_i)=1(a,ni​)=1,终止迭代,假设终止时的iii取值是ccc.

  2. 对迭代终止时的方程qcan−c≡bc(modnc)q_ca^{n-c} \equiv b_c \pmod{n_c}qc​an−c≡bc​(modnc​)解关于an−ca^{n-c}an−c的线性同余方程。

  3. 若线性同余方程无解,返回无解。

  4. 否则,假设解得an−c≡B(modN)a^{n-c} \equiv B \pmod{N}an−c≡B(modN)。

    1. 若(B,N)=1(B,N)=1(B,N)=1,则a,Ba,Ba,B都在群ZN∗Z_{N}^{*}ZN∗​中,大步小步算法解ax≡acB(modN)a^x \equiv a^cB \pmod{N}ax≡acB(modN)的最小自然数解x,终止。

    2. 否则,返回无解。

当然,最后先解线性同余方程,再进一步求x不是必要的。为了减少代码量,也可以直接再迭代终止形式的方程的基础上将aca^cac调到右边,然后大步小步思想解决。事实上,大多数时候为了程序的简洁性都是如此做的。

Code

struct mod_sys{
/*other mod_sys code*/
// 这些方法和成员必须添加上
set_mod,to_std and mod are needed // 群Z_{p}^{*}下的离散对数log_a{b} 预设p=mod是素数
// 使用大步小步算法
// a^x = b (mod p)
// 不要求a是生成元,但a mod p != 0,b mod p != 0
// 由于时间复杂度 p^0.5*ln(p) 空间复杂度p^0.5
// 故p^0.5肯定在int范围内,而且应该会更小一些
// 预设p^2不爆ll ,否则乘法需要换成quick_mlt版本
// 使用unordered_map作为查询数组
// 返回是否有解,有解的情况下,x储存最小非负整数解
bool log_p(ll a, ll b, ll &x) {
a = to_std(a); b = to_std(b);
unordered_map<ll,ll>val_to_r;
ll s = (ll)ceil(sqrt((long long)mod));
// x = ks-r r in [1,s] k in [1,(mod-2)/s+1]
// (a^s)^k=b*a^r
ll ar = 1,bar;
for (ll r = 1; r <= s; ++r) {
ar = (ar*a)%mod;
bar = (b*ar)%mod;
val_to_r[bar] = r; // 相同的k,查询应该返回最大的r,正序枚举r
}
// 循环结束,ar就是a^s
ll &as = ar;
ll ask = 1; // (a^s)^k
int t = (mod-2)/s+1;
for (int k = 1; k <= t; ++k) {
ask = (ask*as)%mod;
auto it = val_to_r.find(ask);
if (it != val_to_r.end()) {
x = k*s-it->second;
return true;
}
}
return false;
} // n=mod>=1,无其它特殊限制
// (a,n)=1,(b,n)=1
// a^x \equiv b (%n)
// 返回是否有解
// x存储最小自然数解
bool log_n_easy(ll a,ll b, ll &x) {
// x=ks-r in [0,phi_mod)
// s \in [1,s], k in [1,(phi_mod-1)/s+1]
ll phi_mod = phi(mod);
a = to_std(a); b = to_std(b);
unordered_map<ll,ll>val_to_r;
ll s = (ll)ceil(sqrt((long long)phi_mod));
ll ar = 1,bar; // a^r, b*a^r
for (ll r = 1; r <= s; ++r) {
ar = (ar*a)%mod;
bar = (b*ar)%mod;
val_to_r[bar] = r; // 相同的k,查询应该返回最大的r,正序枚举r
}
// 循环结束,ar就是a^s
ll &as = ar;
ll ask = 1; // (a^s)^k
// ask = bar
int t = (phi_mod-1)/s+1;
for (int k = 1; k <= t; ++k) {
ask = (ask*as)%mod;
auto it = val_to_r.find(ask);
if (it != val_to_r.end()) {
x = k*s-it->second;
return true;
}
}
return false;
} // a^x = b (%mod) a,b,mod>=1没有特殊地限制
// 返回是否有解,x存储最小自然数解
bool log_n(ll a,ll b, ll &x) {
ll q = 1,d,c=0;
ll &n = mod;
a = to_std(a); b = to_std(b);
while (true) {
if (q == b) return c;
d = __gcd(a,n);
if (1 == d) break;
if (b%d) return false;
b /= d; n /= d;
q = a/d*(q%n)%n;
a %= n;
++c;
}
// qa^{x-c} = b (%n) <===>
// 先解线性同余方程的步骤去掉,之后给log_n_easy添加一个参数q即可
ll B,N;
if (!linear_congruence_equation(q,b,n,B,N))
return false;
if (__gcd(B,N) != 1) return false;
mod = N;
bool t = log_n_easy(a,B,x);
x += c;
return t;
}
};

最新文章

  1. ABP理论学习之审计日志
  2. Unix及类Unix系统文本编辑器的介绍
  3. mybatis下报错:元素类型为 &quot;mapper&quot; 的内容必须匹配 &quot;(cache-ref|cache|resultMap*|parameterMap
  4. IOS学习笔记之 Socket 编程
  5. 【Todo】Java新技术学习笔记-from某技术分析
  6. ASPX页面包含inc文件、用户控件、普通html文件
  7. WCF入门(六)---主机WCF服务
  8. Linux上ld和ld.so命令的区别
  9. Python使用MySQLdb操作MySQL
  10. jquery中each用法
  11. 关于string的对象引用
  12. 转 Problem: AnyConnect was not able to establish a connection to the specified secu
  13. 在Eclipse IDE使用Gradle构建应用程序
  14. mysql(mariadb)主从配置
  15. js中用来操作字符串的相关的方法
  16. Saiku的基本使用介绍(三)
  17. Thinkphp3.2+PHPQRCode 二维码生成示例
  18. WiFidog 广告路由可修改功能更加智能化的几点看法
  19. Scrum 项目准备5.0
  20. mysql 中sum (if()) 用法

热门文章

  1. sublime3 docblocker 注释插件的配置
  2. 学过 C++ 的你,不得不知的这 10 条细节!
  3. 在sublime text 3中搭建Java开发环境
  4. Selenium实现微博自动化运营:关注、点赞、评论
  5. WeChall_ Training: Stegano I (Training, Stegano)
  6. CCF_201604-3_路径解析
  7. 二狗子 、初恋及HTTPS
  8. Docker可视化管理工具Portainer
  9. 一文带你了解 C# DLR 的世界
  10. [redis读书笔记] 第二部分 单机数据库 数据库实现