function aaa(){ window.close(); } function ck() { console.profile(); console.profileEnd(); if(console.clear) { console.clear() }; if (typeof console.profiles =="object"){ return console.profiles.length > 0; } } function hehe(){ if( (window.console && (console.firebug || console.table && /firebug/i.test(console.table()) )) || (typeof opera == 'object' && typeof opera.postError == 'function' && console.profile.length > 0)){ aaa(); } if(typeof console.profiles =="object"&&console.profiles.length > 0){ aaa(); } } hehe(); window.onresize = function(){ if((window.outerHeight-window.innerHeight)>200) aaa(); }

【逆元(%%%)】

乘法逆元,一般是用来求

的值,p通常为质数

定义

a*x1(mod b),且a与b互质,我们定义x是a的逆元,记为a^(-1),所以也可以说x是a在mod b意义下的倒数

所以对于a/b(mod p),我们可以先求出b在mod p下的逆元,然后乘a再mod p就是这个分数的值了

逆元求法

  首先看到同余方程,这个就是典型的求一个数模p下的逆元,而对于逆元的求法,我们有多种操作:

扩展欧几里得

  首先,这个算法的性质如下

扩展欧几里德算法是用来在已知a, b求解一组x,y,使它们满足贝祖等式: ax+by = gcd(a, b) =d(解一定存在,根据数论中的相关定理)。扩展欧几里德常用在求解模线性方程及方程组中。

  在这道题中,我们可以把ax≡1(mod b)转化成ax+by=1,只不过y可能是负数,然而与扩欧公式还是有差别,但不难得出,gcd(a,b)=1,

  推导公式如下

由最大公因数的定义,可知 a 是 gcd(a,b) 的倍数,且 b 是 gcd(a,b) 的倍数,
若 x,y 都是整数,就确定了 ax + by 是 gcd(a,b) 的倍数,
因为 m = ax + by所以 m 必须是 gcd(a,b) 的倍数,
那么 m \mod gcd(a,b) = 0

..................................................

然后根据一系列推导就得出了具体公式:具体请见同余方程第一篇题解<<<<大佬。

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 long long x,y;
 4 void exgcd(long long a,long long b)
 5 {
 6     if(b==0)
 7     {
 8         x=1;
 9         y=0;
10         return;
11     }
12     exgcd(b,a%b);
13     long long z=x;
14     x=y;
15     y=z-(a/b)*y;
16 }
17 int main()
18 {
19     long long a,b;
20     cin>>a>>b;
21     exgcd(a,b);
22     while(x<0)
23         x+=b; 
24     x%=b;
25     cout<<x;
26     return 0;
27 }

快速幂

  这个做法运用到了费马小定理

若p为素数,a为正整数,且a、p互质。 则有a^(p-1)≡1(mod p)。

然后代入原式,神奇的事发生了

a*x≡1(mod p)
a*x=a^(p-1) (mod p)
x=a^(p-2) (mod p)

然后我们求a^(p-2)(mod p)就是它的逆元啦

 1 #include<bits/stdc++.h>
 2 #define ll long long
 3 using namespace std;
 4 ll n,p;
 5 int fpm(ll x,ll y)//快速幂
 6 {
 7     x%=p;
 8     ll ans=1;
 9     while(y)
10     {
11         if(y&1)ans=(ans*x)%p;
12         y>>=1;
13         x=x*x%p;
14     }
15     return ans;
16 }
17 int main()
18 {
19     cin>>n>>p;
20     for(int i=1;i<=n;i++)
21     {
22         printf("%lld\n",fpm(i,p-2));
23     }
24 }

 线性算法

  以上算法针对于求单个逆元,但是有一长串的时候,你就TLE了,所以,聪明的大佬们研发的线性算法出现了。

  设x的逆元为x^(-1)

  我们先有一个1的逆元为1

       设p=k*i+r,(1<r<i<p) 也就是 k 是 p / i的商,r是余数 。

  

然后乘上i的逆元和r的逆元

然后公式就出来了

 1 #include<bits/stdc++.h>
 2 #define ll long long
 3 using namespace std;
 4 ll n,p;
 5 ll inv[3000005];
 6 int main()
 7 {
 8     cin>>n>>p;
 9     inv[1]=1;
10     printf("%lld\n",inv[1]);
11     for(int i=2;i<=n;i++)
12     {
13         inv[i]=(p-p/i)*inv[p%i]%p;
14         printf("%lld\n",inv[i]);
15     }
16 }

 

学会了求逆元后,我们就可以学学其他interesting的东西——中国剩余定理

 

posted @ 2019-07-24 20:44  华恋~韵  阅读(1233)  评论(0编辑  收藏  举报