目录:
1.逆元的作用
2.逆元的定义
3.单个逆元的求法
4.多个逆元的求法
先知道是干什么的,能解决什么问题
我所知道的数论题中常见的出现模运算
( a + b ) % m o d = ( a % m o d + b % m o d ) % m o d (a+b)\%mod=(a\%mod+b\%mod)\%mod (a+b)%mod=(a%mod+b%mod)%mod
( a ∗ b ) % m o d = ( a % m o d ∗ b % m o d ) % m o d (a*b)\%mod=(a\%mod*b\%mod)\%mod (a∗b)%mod=(a%mod∗b%mod)%mod
现在给出 ( 3 ∗ 6 / 3 ) m o d 7 (3*6/3)mod 7 (3∗6/3)mod7
第一种算法:原式= ( 18 / 3 ) m o d 7 (18/3)mod7 (18/3)mod7=6;
第二种算法:原式= ( ( 18 m o d 7 ) / 3 ) m o d 7 = ( 4 / 3 ) m o d 7 = 1 ? × ((18mod7)/3)mod7=(4/3)mod7=1?× ((18mod7)/3)mod7=(4/3)mod7=1?×
显然如果计算机处理的话结果肯定是1,也就是说除法不存在取模的运算法则,在某些时候就会产生数字太大精度错误等问题。
化除为乘是一个好的处理方法,记得以前做过一个题目,排序两个结构体使a/b的值大的排前面,如果写return x 1 . a / x 1 . b > x 2 . a / x 2 . b x_1.a/x_1.b>x_2.a/x_2.b x1.a/x1.b>x2.a/x2.b就会出错,但是写成return x 1 . a × x 2 . b > x 1 . b × x 2 . a x_1.a×x_2.b>x_1.b×x_2.a x1.a×x2.b>x1.b×x2.a就可以了。那么这里我们能不能找到一个整数X满足 ( 4 / 3 ) m o d 7 = ( 4 ∗ X ) m o d 7 (4/3)mod7=(4*X)mod7 (4/3)mod7=(4∗X)mod7呢?如果可以的话就会把问题简化很多了
从这个定义可以看出,如果a,p不互质的话,也就不存在a关于模p的乘法逆元!
由定义可知 a x = p y + 1 ax=py+1 ax=py+1移项可得
a x + m p = 1 ( a , p 互 质 且 已 知 ) ax+mp=1(a,p互质且已知) ax+mp=1(a,p互质且已知)
那么显然第一种求逆元的方法——exgcd求解方程 a x + m p = 1 ax+mp=1 ax+mp=1的整数 x x x即为a关于模p的一个逆元(逆元数量不是唯一的)
每求一次逆元的复杂度是 O ( l o g ( a + p ) ) O(log(a+p)) O(log(a+p))
ll exgcd(ll a,ll b,ll &x,ll &y)
{
if (b==0)
{
x=1;
y=0;
return a;
}
ll gcd=exgcd(b,a%b,y,x);
y-=(a/b)*x;
return gcd;
}
int main()
{
ll g=exgcd(a,b,x,y);//注意输入时a,p对应a,b
if (g!=1){NO;}
ll t=b/1;
if (t<0)t=-t;
ll inv=(x%t+t)%t;
}
当p为素数的时候才可以用!!!!!
a p − 1 ≡ 1 a^{p-1}\equiv 1 ap−1≡1 m o d mod mod p p p ===> a × a p − 2 ≡ 1 a×a^{p-2}\equiv 1 a×ap−2≡1 m o d mod mod p p p
这不就是定义吗,所以 a p − 2 a^{p-2} ap−2就是a模p意义下的逆元,快速幂求一次即可
每求一次逆元的复杂度是 O ( l o g ( p − 2 ) ) O(log(p-2)) O(log(p−2)),优于exgcd但局限于质数范围
ll qpow(ll a,ll b,ll mod)
{
ll res=1;
while(b)
{
if (b&1)res=(res*a)%mod;
a=(a*a)%mod;
b>>=1;
}
return res%mod;
}
int main()
{
WW(qpow(a,p-2,p));
}
i n v [ i ] inv[i] inv[i]表示i在模p意义下的逆元
主要用到递推式: i n v [ i ] = ( p − p / i ) ∗ i n v [ p % i ] % p inv[i]=(p-p/i)*inv[p\%i]\%p inv[i]=(p−p/i)∗inv[p%i]%p
大佬证明(本人菜鸡不会):
设 p = k ∗ i + r p=k*i+r p=k∗i+r===> k ∗ i + r ≡ 0 k*i+r\equiv 0 k∗i+r≡0 m o d mod mod p p p
两边同乘 i − 1 r − 1 i^{-1}r^{-1} i−1r−1,得到
k ∗ r − 1 + i − 1 ≡ 0 k*r^{-1}+i^{-1}\equiv 0 k∗r−1+i−1≡0 m o d mod mod p p p
整理得到
i − 1 ≡ − ⌊ p / i ⌋ ∗ ( p % i ) − 1 i^{-1}\equiv -\lfloor p/i \rfloor*(p\%i)^{-1} i−1≡−⌊p/i⌋∗(p%i)−1 m o d mod mod p p p,证明完毕
由于这样求出来的 i n v [ i ] inv[i] inv[i]可能是负数,根据性质
a ≡ b a\equiv b a≡b m o d mod mod m m m
则 a ≡ b + m k a\equiv b+mk a≡b+mk m o d mod mod m m m
对右边 + p ∗ i n v [ p % i ] +p*inv[p\%i] +p∗inv[p%i];
综上所述,代码如下:
int main()
{
ll n,p;
scanf("%lld%lld",&n,&p);
inv[1]=1;
rep(i,2,n)
{
inv[i]=(p-p/i)*inv[p%i]%p;
}
rep(i,1,n)WW(inv[i]);
}
放一道差点错了的题
这就是最明显的直接套用逆元求解大数除法mod
用费马小定理也可以解……
/**
* Author1: low-equipped w_udixixi
* Author2: Sher丶lock
* Date :2019-09-10
**/
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define ll long long
#define pb push_back
#define rep(x,a,b) for (int x=a;x<=b;x++)
#define repp(x,a,b) for (int x=a;x
#define W(x) printf("%d\n",x)
#define WW(x) printf("%lld\n",x)
#define pi 3.14159265358979323846
#define mem(a,x) memset(a,x,sizeof a)
#define lson rt<<1,l,mid
#define rson rt<<1|1,mid+1,r
using namespace std;
const int maxn=2e6+7;
const int INF=1e9;
const ll INFF=1e18;
const ll MOD=19260817;
char s1[maxn];
char s2[maxn];
ll exgcd(ll a,ll b,ll &x,ll &y)
{
if (b==0)
{
x=1;
y=0;
return a;
}
ll gcd=exgcd(b,a%b,y,x);
y-=x*(a/b);
return gcd;
}
int main()
{
ll a=0,b=0,x,y;
cin>>s1>>s2;int l1=strlen(s1),l2=strlen(s2);
rep(i,0,l1-1)
a=(a*10+s1[i]-'0')%MOD;
rep(i,0,l2-1)
b=(b*10+s2[i]-'0')%MOD;
ll g=exgcd(b,MOD,x,y);
if (g!=1){cout<<"Angry!"<<endl;return 0;}
ll t=MOD/1;
if (t<0)t=-t;
ll inv=(x%t+t)%t;
ll ans=a*inv%MOD;
WW(ans);
}
模板题(只会模板题的wudixixi)
直接套费马小定理求解
/**
* Author1: low-equipped w_udixixi
* Author2: Sher丶lock
* Date :2019-09-10
**/
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define ll long long
#define pb push_back
#define rep(x,a,b) for (int x=a;x<=b;x++)
#define repp(x,a,b) for (int x=a;x
#define W(x) printf("%d\n",x)
#define WW(x) printf("%lld\n",x)
#define pi 3.14159265358979323846
#define mem(a,x) memset(a,x,sizeof a)
#define lson rt<<1,l,mid
#define rson rt<<1|1,mid+1,r
using namespace std;
const int maxn=2e6+7;
const int INF=1e9;
const ll INFF=1e18;
ll qpow(ll a,ll b,ll mod)
{
ll res=1;
while(b)
{
if (b&1)res=(res*a)%mod;
a=(a*a)%mod;
b>>=1;
}
return res%mod;
}
int main()
{
int t,a,b;
scanf("%d",&t);
while(t--)
{
scanf("%d%d",&a,&b);
int inv=qpow(b,9971,9973);
W(a*inv%9973);
}
return 0;
}