zoukankan      html  css  js  c++  java
  • 逆元&欧拉函数

    欧拉函数:

      φ(p)表示小于p的正整数中与p互质的数的个数,称作欧拉函数。

      求单个数的欧拉函数时可以利用来求

      其中pi为p分解出的质因数,ki表示该质因数的指数

      代码:

    #include<cstdio>
    #include<iostream>
    using namespace std;
    int phi[100];
    int Eurl(int x)
    {
        int ans=1;
        for(int i=2;i*i<=x;++i)
        {
            if(x%i==0)
            {
                x/=i;
                ans*=i-1;
            }
            while(x%i==0)
            {
                x/=i;
                ans*=i;
            }
        }
        if(x!=1) ans*=(x-1);
        return ans;
    }
    int main()
    {
        int n=1;
        while(1)
        {
            scanf("%d",&n);
            if(!n) break;
             printf("%d
    ",Eurl(n));
        }
        return 0;
    }
    单个欧拉函数

      还可以求范围内的欧拉函数

      代码:

    #include<cstdio>
    #include<iostream>
    using namespace std;
    int phi[100];
    int main()
    {
        int l,r;
        scanf("%d%d",&l,&r);
        for(int i=1;i<=r;++i)
        phi[i]=i;
        for(int i=2;i<=r+1;++i)
        {
            if(phi[i]==i)
            {
                for(int j=i;j<=r;j+=i)
                {
                    phi[j]=phi[j]/i*(i-1);
                    
                }
                
            }
        }
        for(int i=l;i<=r;++i)
        {
            printf("%d %d
    ",i,phi[i]);
        }
        return 0;
    }
    范围内欧拉函数

    逆元:

      a*a-1≡1(mod p)

      a-1叫做a在mod p 意义下的逆元。

      利用欧拉定理可知

      aφ(p)≡1(mod p)

      a*aφ(p)-1≡1(mod p)

       所以a在mop p意义下的逆元就是aφ(p)-1

       求单个逆元可以直接这样求

      一种O(n)求范围内逆元的办法:

        inv[i]=(p-p/i)*inv[p%i]%p

      证明:

        设t=p/i   k=p%i

        易知p≡0(mod p)

        所以t*i+k≡0(mod p)

        t*i≡-k(mod p)

        -t*i≡k(mod p)

        两边都乘以inv[i]和inv[k]

        -t*i*inv[i]*inv[k]≡k*inv[k]*inv[i](mod p)

        -t*inv[k]≡inv[i](mod p)

        所以就得到了

        inv[i]≡(-p/i)*inv[p%i]  (mod p)

        所以inv[i]=(p-p/i)*inv[p%i]%p

        用p-p/i是为了避免出现负数

      代码:

     1 #include<cstdio>
     2 #include<iostream>
     3 using namespace std;
     4 int inv[110];
     5 int main()
     6 {
     7     int n,p;
     8     scanf("%d%d",&n,&p);
     9     inv[1]=1;
    10     for(int i=2;i<=n;++i)
    11         inv[i]=inv[p%i]*(p-p/i)%p;
    12     for(int i=1;i<=n;++i)
    13         printf("%d ",inv[i]);
    14     return 0;
    15 }
    范围内逆元

       一种求从1到n间阶乘的逆元的方法

        依据逆元的性质易知

          n!*n!-1≡(n-1)!*(n-1)!-1≡1(mod p)

        即 n*(n-1)!*n!-1≡(n-1)!*(n-1)!-1≡1(mod p)

        然后两边同时乘以(n-1)!-1得到

          n*n!-1≡(n-1)!-1(mod p)

        这样就可以倒着递推出范围内阶乘的逆元了

        代码:

     1 #include<cstdio>
     2 #include<iostream>
     3 using namespace std;
     4 int jsinv[110];
     5 int fai(int k)
     6 {
     7     int ans=1;
     8     for(int i=2;i*i<=k;++i)
     9     {
    10         if(k%i==0)
    11         {
    12             ans*=i-1;
    13             k/=i;
    14         }
    15         while(k%i==0)
    16         {
    17             ans*=i;
    18             k/=i;
    19         }
    20     }
    21     ans*=k;
    22     return ans;
    23 }
    24 int mi(int a,int k,int p)
    25 {
    26     int ans=1;
    27     for(int now=a;k;k>>=1,now=now*now%p)
    28         if(k&1) ans=ans*now%p;
    29     return ans;
    30 }
    31 int main()
    32 {
    33     int n,p;
    34     scanf("%d%d",&n,&p);
    35     int k=1;
    36     for(int i=2;i<=n;++i)
    37         k=k*i%p;
    38     jsinv[n]=mi(k,fai(p),p);
    39     for(int i=n-1;i>=1;--i)
    40         jsinv[i]=(jsinv[i+1]*(i+1))%p;
    41     for(int i=1;i<=n;++i)
    42         printf("%d ",jsinv[i]);
    43     return 0;
    44 }
    范围内阶乘的逆元
  • 相关阅读:
    Codeforces Round #579 (Div. 3)
    2019牛客暑期多校训练营(第一场)ABBA
    Comet OJ
    Codeforces Round #562 (Div. 2)
    Codeforces Round #569 (Div. 2)
    寒假集训总结
    线段树做题总结
    Git学习
    unity开发vuforia血的教训
    (学习13)最短圆排列问题
  • 原文地址:https://www.cnblogs.com/wxyww/p/9188788.html
Copyright © 2011-2022 走看看