乘法逆元及逆元求法

前置知識

模運算:取餘運算,即a除以b獲得的餘數,記爲mod,又記爲%c++

模運算過程當中,加減乘均可以先對a,b進行%p,而後再進行加減乘,最後再%p,結果不變ui

運算符優先級,模運算和乘除法的運算符優先級是同樣的spa

同餘:a和b除以p獲得的餘數相同,即p能夠整除(a-b)code

乘法逆元的問題背景

求解\(\cfrac{a}{b} \mod p\)的值,由於除法不能直接對a,b先取模再進相除,因此這邊就引入了逆元get

逆元,能夠理解爲是在mod p意義下b的倒數,下文中將b的逆元記錄爲inv[b]
如下是逆元的定義
\(b \times inv[b] \equiv 1 \pmod p\)it

逆元的性質及其證實

1.第一個性質
\(\cfrac{a}{b} \equiv a \times inv[b] \pmod p\)
證實以下
\(b \times inv[b] \equiv 1 \pmod p\)
\((b \times inv[b]-1)\bmod p=0\)
由模運算的乘法性質\(a \times b \bmod p=(a \bmod p) \times (b \bmod p)\bmod p\)
獲得\(\cfrac{a}{b}\times(b \times inv[b]-1)\bmod p=0\)
\((a \times inv[b]- \cfrac{a}{b})\bmod p=0\)
\(a \times inv[b] \equiv \cfrac{a}{b} \pmod p\)
2.惟一性
設b有兩個逆元c,d
\(a*c \equiv a*d \pmod p\)
\((a*c-a*d)\pmod p=0\)
\(a(c-d) \mod p=0\)
\(a\% p \neq 0\)
由模運算的乘性質獲得\(c=d\)
因此b的逆元惟一
3.可積性
\(inv[a] \times inv[b]=inv[a \times b]\)
證實
\(a*b * inv[ab]\equiv \pmod p\)
又由於\(a*inv[a]\equiv 1\)
\(b*inv[b]\equiv 1\)
模運算乘性質易證可積性成立
4.週期性
求解\(\cfrac{a}{b} \mod p\)中b的逆元
通常默認b小於p,可是當b大於p的時候來求逆元,其實就是一個週期性
\(inv[k*p+r]=inv[r],0<r<p,k \in N^*\)
證實
由於同餘,知足兩邊同乘以一個數依舊成立
因此有如下
\(\cfrac{n}{r} \equiv k_1 \pmod p\)
\(\cfrac{n}{tp+r} \equiv k_2 \pmod p\)
則對於第二個式子,兩邊同時乘以\(tp+r\)
\(n \equiv k_2r \pmod p\)
\(\cfrac{n}{r} \equiv k_2 \pmod p\)
同理\(\cfrac{n}{r} \equiv k_1 \pmod p\)
\(k_1 \equiv k_2 \pmod p\)
\(b=r,c=t*p+r\),則有
\(\cfrac{n}{b} \equiv \cfrac{n}{c} \pmod p\)
\(n*inv[b] \equiv n*inv[c] \pmod p\)
\(inv[b] \equiv inv[c] \pmod p\)
又由於\(0\le inv[b],inv[c] <p\)
因此\(inv[b]=inv[c]\)class

逆元求法

費馬小定理和拓展歐幾里得這邊就不講了
這邊主要講線性求逆元
一下模的結果範圍爲[0,r-1]
\(p=k*i+r\)
\(k*i+r \equiv 0 \pmod p\)
模運算乘性質有
式子兩邊同時乘以\(inv[i]*inv[r]\)
\(k*inv[r]+inv[i] \equiv 0 \pmod p\)
\(inv[i] \equiv k*inv[r] \pmod p\)
\(inv[i] \equiv -\cfrac{p}{i}*inv[p \% i] \pmod p\)
其中p%i在[0,i-1]範圍內,即求i的時候p%i的逆元已經被求出
1.線性求逆元
原題見,洛谷P3811乘法逆元模版題im

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int mx=20000999;
ll inv[mx];
void getinv(ll n,ll p){
	inv[1]=1;
	for(int i=2;i<=n;i++){
		inv[i]=(1ll*(p-(p/i))*inv[p%i])%p;
		//由於C++對於負數取模仍是負數因此須要+p 
		//中間相乘可能會爆int範圍,因此用ll 
	}
}
int main(){
	int n,p;
	scanf("%d%d",&n,&p); 
	getinv(n,p);
	for(int i=1;i<=n;i++){
		printf("%d\n",inv[i]);
	}
}

2.線性求單個逆元模版

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int mx=20000999;
ll inv[mx];
ll getinv(int x,int p) {
	return x==1?1:1ll*(p-p/x)*getinv(p%x,p)%p;
}
int main() {
	int n,p;
	scanf("%d%d",&n,&p);
	cout<<getinv(n,p);
}
相關文章
相關標籤/搜索