通常我很懒,计算模逆如下:
def inv(a,m):
return 1 if a==1 else ((a-inv(m%a,a))*m+1)//a
print(inv(100**7000,99**7001))
但我很想知道将更多信息传回的方法,即 Bezout 定理的解决方案(而不仅仅是其中一个),在实践中是否会产生更快或更慢的算法:
def bez(a,b):
# returns [x,y,d] where a*x+b*y = d = gcd(a,b) since (b%a)*y+a*(x+(b//a)*y)=d
if a==0: return [0,1,b] if b>0 else [0,-1,-b]
r=bez(b%a,a)
return [r[1]-(b//a)*r[0],r[0],r[2]]
def inv(a,m):
r=bez(a,m)
if r[2]!=1: return None
return r[0] if r[0]>=0 else r[0]+abs(m)
print(inv(100**7000,99**7001))
我惊奇地发现后者的运行速度比前者快了 50 多倍!但是两者都使用几乎相同数量的递归调用和每次调用 1 个整数除法和 1 个模运算,并且操作数的位长度平均大约是前者的两倍(因为所涉及的参数是相同的),所以我只期望它的时间复杂度大约是后者的 4 倍,而不是 50 倍。
那么为什么我会观察到这种奇怪的加速呢?
请注意,我使用的是 Python 3,并且在online运行时观察到了这一点,并在顶部添加了以下代码以阻止 Python 解释器抱怨超出最大递归深度或堆栈空间:
import resource,sys
sys.setrecursionlimit(1000000)
resource.setrlimit(resource.RLIMIT_STACK,[0x10000000,resource.RLIM_INFINITY])