CRT合并

对于两个同余方程组:

x mod a1 = b1 (1)

x mod a2 = b2 (2)

我们令x = a1 * x0 + b1;

带入(2)式,即(a1* x0 + b1) mod a2 = b2,将b1移项,即:

a1 * x0 mod a2 = b2 – b1 (3)

显然,我们可以使用ex_gcd求出它的一个可行解x1,而实际上x0满足通项:

x0 = x1 + a2 / gcd(a1, a2) * k

带回(3)式可得

a1 * x1 + lcm(a1, a2) * k + P = 0

其中a1 * x1 + P是个常数,也就是说,我们将两个同余方程组合并成了一个方程组:

x mod lcm(a1, a2) = a1 * x1 + P

所以,我们只需不断合并,就可以将原方程组化为一个方程,就可以直接求出一组可行解了。

void CRT(int64 a1, int64 b1, int64 a2, int64 b2, int64 &a, int64 &b) {
	if (b1 > b2) swap(b1, b2), swap(a1, a2);
	int64 x, y, g;
	g = gcd(a1, a2); a1 /= g, a2 /= g;
	ex_gcd(a1, a2, x, y);

	a = a1 * a2 / g;
	x = x * g % a2 * (b2 - b1) / g % a;
	b = ((b1 + a1 * x) % a + a) % a;
}

有一些地方比较容易爆long long,需要小心。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值