中国剩余定理

中国剩余定理用来求解同余方程组。其中 \(m_i\) 两两互质

\( \begin{cases} x &\equiv a_1 \pmod {m_1}\\ x &\equiv a_2 \pmod{m_2}\\ &\vdots\\ x & \equiv a_k\pmod {m_k}\\ \end{cases} \)

定义 \(M=\prod\limits_{i=1}^km_i;\ c_i=M_i/m_i\)\(c_i^{-1}\) 是模 \(m_i\) 意义下的逆元

\(x=\sum\limits_{i=1}^k{a_ic_ic_i^{-1}}\pmod M\)

不要以为这里的 \(c_ic_i^{-1}=1\) 模数不一样的。

如果想深入理解,我们需要知道公式是怎么来的。

我们可以参考:https://zhuanlan.zhihu.com/p/103394468https://www.bilibili.com/video/BV1AN4y1N7Su/?spm_id_from=333.999.0.0&vd_source=700dde1e30c453844bbbd11aff30a401

我们先感性思考下,想使得 \(x\) 满足这么多方程式不容易,但是如果是……

\( \begin{cases} n_1 &\equiv a_1 \pmod {m_1}\\ n_2 &\equiv a_2 \pmod{m_2}\\ &\vdots\\ n_k & \equiv a_k\pmod {m_k}\\ \end{cases} \)

\(n_i\) 去满足对应的一个方程,是不是会更简单些?那么这和 \(x\) 有什么关系?难不成将 \(n_i\) 全加起来?显然不行,比如说我们考虑:

\( \begin{cases} n_1 &\equiv 2 \pmod {3}\\ n_2 &\equiv 3 \pmod{5}\\ \end{cases} \)

显然不是随意的 \(n_1+n_2\),如果 \(x=n_1+n_2\) 的话,分别让其满足一式和二式,则 \(x\) 分别得是 \(3\)\(5\) 的倍数。我们扩展下,来到经典例子:

\( \begin{cases} n_1 &\equiv 2 \pmod {3}\\ n_2 &\equiv 3 \pmod{5}\\ n_3 &\equiv 2 \pmod{7}\\ \end{cases} \)

运用刚才的思路,应 \(n_1\)\(35\) 的倍数,\(n_2\)\(21\) 的倍数,\(n_3\)\(15\) 的倍数。

于是方程化为:

\( \begin{cases} 35w_1 &\equiv 2 \pmod {3}\\ 21w_2 &\equiv 3 \pmod{5}\\ 15w_3 &\equiv 2 \pmod{7}\\ \end{cases} \)

然后发现对于单个同余方程,可以 exgcd 解决。


具体求解步骤

  • 计算模数的积 \(M\)
  • 计算第 \(i\) 个方程的 \(c_i=\frac{M}{m_i}\)
  • 计算 \(c_i\) 在模 \(m_i\) 意义下的逆元 \(c_i^{-1}\)
  • \(x=\sum\limits_{i=1}^na_ic_ic_i^{-1}\pmod M\)

证明公式满足

证明 \(x=\sum\limits_{i=1}^na_ic_ic_i^{-1}\pmod M\) 满足每个同余方程

因为模数两两互质,又 \(c_i=\frac{M}{m_i}\),即除去了 \(m_i\),即剩下的乘积一定都与 \(m_i\) 互质,所以 \(c_i \perp m_i\),互质才会有逆元。

\(i \neq j\) 时,\(c_j\equiv 0\pmod {m_i}\),则 \(c_jc_j^{-1}\equiv 0\pmod {m_i}\)

\(i = j\) 时,\(c_j\not\equiv 0\pmod {m_i}\),则 \(c_jc_j^{-1}\equiv 1\pmod {m_i}\)

当公式中的 \(i\ne j\) 的时候,数值都为 \(0\),都消失了,只留下了 \(i=j\) 时的 \(1\)

\(x\equiv\sum\limits_{j=1}^na_jc_jc_j^{-1}\pmod {m_i}\\ \equiv r_ic_ic_i^{-1}\pmod {m_i}\\ \equiv r_i\pmod {m_i}\)

代码+模板题两道

inline ll CRT(ll a[], ll b[], ll n)
{
    ll p = 1, x = 0;
    for (int i = 0; i < n; ++i)
        p *= a[i];
    for (int i = 0; i < n; ++i)
    {
        ll r = p / a[i];
        x += (b[i] * r * inv(r, a[i])) % p; // inv函数是求逆元,可以用exgcd实现,注意这里不能用费马小定理呢
    }
    return x % p;
}

板子,不会的话记下来


模板题,洛谷P1495 曹冲养猪

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
ll exgcd(ll a, ll b, ll &x, ll &y)
{
    if (b == 0)
    {
        x = 1;
        y = 0;
        return a;
    }
    ll d = exgcd(b, a % b, y, x);
    y -= (a / b) * x;
    return d;
}
inline ll inv(ll a, ll p)
{
    ll x, y;
    exgcd(a, p, x, y);
    return (x % p + p) % p;
}
inline ll CRT(ll a[], ll b[], ll n)
{
    ll p = 1, x = 0;
    for (int i = 0; i < n; ++i)
        p *= a[i];
    for (int i = 0; i < n; ++i)
    {
        ll r = p / a[i];
        x += (b[i] * r * inv(r, a[i])) % p;
    }
    return x % p;
}
int main()
{
    ll n, a[10], b[10];
    scanf("%lld", &n);
    for (int i = 0; i < n; ++i)
        scanf("%lld%lld", a + i, b + i);
    printf("%lld\n", CRT(a, b, n));
    return 0;
}

P3868 [TJOI2009] 猜数字

这道题略加转换就变成了模板题

\(\forall i\in [1,k]\),有 \(b_i | (n-a_i)\)

根据同余基本性质:

\(n-a_i\equiv 0\pmod {b_i}\)

\(n\equiv a_i\pmod {b_i}\)

用中国剩余定理解就好了,都说了bi两两互质。

#include <bits/stdc++.h>
#define rei register int
#define LL long long
#define IOS ios::sync_with_stdio(false); cin.tie(0); cout.tie(0);
#define cvar int n, m, T;
#define rep(i, s, n, c) for (register int i = s; i <= n; i+=c)
#define repd(i, s, n, c) for (register int i = s; i >= n; i-=c)
#define CHECK cout<<"WALKED"<<endl;
inline int read(){int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();} while(ch>='0' && ch<='9')x=(x<<3)+(x<<1)+ch-'0',ch=getchar();return x*f;}
#define pb push_back
#define ls id<<1
#define rs id<<1|1
const int INF = INT_MAX;
long long binpow(long long a, long long b, LL mod){long long res = 1;  while (b > 0){if (b & 1) res = res * a % mod;a = a * a % mod;  b >>= 1;  }  return res;}
using namespace std;

__int128 read128()
{
	__int128 f = 1, w = 0;
	char ch = getchar();
	while (ch < '0' || ch > '9') {
		if (ch == '-')
			f = -1;
		ch = getchar();
	}
	while (ch <= '9' && ch >= '0') {
		w = (w << 1) + (w << 3) + ch - '0';
		ch = getchar();
	}
	return f * w;
}
void print(__int128 x)
{
	if (x < 0) {
		putchar('-');
		x = -x;
	}
	if (x > 9) print(x / 10);
	putchar(x % 10 + '0');
}

__int128 a[20], b[20], M = 1;
int K;

__int128 exgcd(__int128 a, __int128 b, __int128 &x, __int128 &y)
{
	if (b == 0)
	{
		x = 1;
		y = 0;
		return a;
	}
	__int128 d = exgcd(b, a % b, y, x);
	y -= ( a/ b) * x;
	return d;
}

__int128 inv(__int128 a, __int128 mod)
{
	__int128 x, y;
	exgcd(a, mod, x, y);
	return (x % mod + mod) % mod;
}

void crt()
{
	__int128 x = 0;
	rep (i, 1, K, 1) {
		__int128 ri = a[i], ci = M / b[i], ci_inv = inv(ci, b[i]);
		x = (x + ri * ci * ci_inv) % M;
	}
	x %= M;
	print(x);
}

int main()
{
	
	cin >> K;
	rep (i, 1, K, 1) a[i] = read128();
	rep (i, 1, K, 1) b[i] = read128();
	rep (i, 1, K, 1) a[i] = (a[i] % b[i] + b[i]) % b[i];
	// b[i]互质,crt即可
	rep (i, 1, K, 1) M = M * b[i];
	crt();
	return 0;
}
posted @ 2022-10-24 06:21  Vegdie  阅读(56)  评论(0编辑  收藏  举报