热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

关于数论的开发笔记

本文由编程笔记#小编整理,主要介绍了关于数论相关的知识,包括数论的算法和百度百科的链接。文章还介绍了欧几里得算法、辗转相除法、gcd、lcm和扩展欧几里得算法的使用方法。此外,文章还提到了数论在求解不定方程、模线性方程和乘法逆元方面的应用。摘要长度:184字。
本文由编程笔记#小编为大家整理,主要介绍了关于数论相关的知识,希望对你有一定的参考价值。


数论东西很多又很杂,所以想要总结一下,有一些算法的百度百科讲得很清楚,所以我就直接给了个链接在这(其实是懒23333),方便自己复习吧。


欧几里得算法

百度百科


辗转相除法求gcd与lcm

使用辗转相除算出gcd后,lcm可以直接通过gcd算出,但是注意求lcm的过程可能爆int,建议使用long long

inline int gcd(int a,int b){while(b^=a^=b^=a%=b);return a;}
inline long long lcm(int a,int b){return (long long)a/gcd(a,b)*b;}

扩展欧几里得算法

百度百科

int x,y;
inline int exgcd(int a,int b,int &x,int &y)
{
if(!b)
{
x=1,y=0;
return a;
}
int gcd=exgcd(b,a%b,x,y);
int t=x;x=y,y=t-a/b*y;
return gcd;
}

应用:



  • 求不定方程:ax+by=c

  • 解模线性方程:ax≡m(mod b) 转化为ax+by=m,与不定方程求解方法相同。

  • 求解乘法逆元

这里多说几句逆元:

定义:存在ax≡1(mod p),则称x是a关于模p的乘法逆元

定理:a关于p的乘法逆元的充要条件是gcd(a,p)=1

用途:(a/b)%p=(a*x)%p

Tip:mod 1的逆元就是1

例题HDU1576

在题目中,我们将B在模9973下的逆元设为x,求(A/B)%9973就可以化为(A×x)%9973,这样的话就可以通过模的性质变成A%9973×x%9973,而在题目中A%9973已经给出,那么只需要通过拓展欧几里得求出逆元之后就可以得到答案了

代码:

#include
#define ll long long
using namespace std;
templateinline void read(TP&x)
{
x=0;int f=1;char c=getchar();
while(c>‘9‘||c<‘0‘){if(c==‘-‘)f=-1;c=getchar();}
while(c>=‘0‘&&c<=‘9‘){x=(x<<1)+(x<<3)+(c^48);c=getchar();}
x*=f;
}
templateinline void print(TP x)
{
if(x<0)x=-x,putchar(‘-‘);
if(x>=10)print(x/10);
putchar(x%10+‘0‘);
}
ll t,a,b,ans;
ll x,y;
inline ll exgcd(ll a,ll b,ll &x,ll &y)
{
if(!b)
{
x=1,y=0;
return a;
}
ll gcd=exgcd(b,a%b,x,y);
ll t=x;x=y,y=t-a/b*y;
return gcd;
}
int main()
{
read(t);
while(t--)
{
read(a),read(b);
exgcd(b,9973,x,y);
x=((x%9973)+9973)%9973;
print(a*x%9973),putchar(‘
‘);
}
}



欧拉筛

利用了每个合数必有一个最小素因子,每个合数仅被它的最小素因子筛去正好一次。

prime数组 中的素数是递增的,当 i 能整除 prime[j],那么 i*prime[j+1] 这个合数肯定被 prime[j] 乘以某个数筛掉。

代码中体现在:
if(i%prime[j]==0)break;

就可以做到几乎是线性的时间复杂度就可以筛出素数

int prime[2000000],size;
bitset<2000000>flag;
inline void make_prime(int x)
{
for(int i=2;i<=x;++i)
{
if(!flag[i]) prime[++size]=i;
for(int j=1;j<=size&&prime[j]*i<=x;++j)
{
flag[prime[j]*i]=1;
if(!(i%prime[j])) break;
}
}
}

费马平方和定理

内容如下:

除2以外,任何素数都能分成4k+1与4k+3的形式,任何形为4k+1的素数都能表示为两个平方数之和,而4k+3形式质数则不能

例题:CF113C double happiness

代码:

#include
#include
#include
#include
#include
#include<iostream>
#define ri register int
#define rc register char
using namespace std;
templateinline void read(TP&x)
{
x=0;int f=1;char c=getchar();
while(c>‘9‘||c<‘0‘){if(c==‘-‘)f=-1;c=getchar();}
while(c>=‘0‘&&c<=‘9‘){x=(x<<1)+(x<<3)+(c^48);c=getchar();}
x*=f;
}
templateinline void write(TP x)
{
if(x==0){putchar(‘0‘);return;}
char f[100]={0};int i=0;
if(x<0){putchar(‘-‘),x=-x;}
while(x){++i,f[i]=x%10+48;x/=10;}
while(i){putchar(f[i]),--i;}
}
int prime[20000000],size;
bitset<300000005>flag;
int l,r,tot;
inline void make_prime(int x)
{
for(register int i=2;i<=x;++i)
{
if(!flag[i]) prime[++size]=i;
for(register int j=1;j<=size&&i*prime[j]<=x;++j)
{
flag[prime[j]*i]=1;
if(!(i%prime[j])) break;
}
}
}
int main()
{
read(l),read(r);
make_prime(r);
for(register int i=1;i<=size;++i)
{
if(prime[i] if(prime[i]%4==1)
++tot;
}
if(l<=2&&r>=2) ++tot;
write(tot);
return 0;
}

求欧拉函数


线性筛法

线性筛法证明,右转某大佬的csdn

int prime[2000000],eular[2000000],size;
bitset<2000000>flag;
inline void Eular(int x)
{
eular[1]=1;
for(int i=2;i<=x;++i)
{
if(!flag[i]) prime[++size]=i,eular[i]=i-1;
for(int j=1;j<=size&&prime[j]*i<=x;++j)
{
flag[prime[j]*i]=1;
if(!(i%prime[j]))
{
eular[prime[j]*i]=eular[i]*prime[j];
break;
}
eular[prime[j]*i]=eular[i]*eular[prime[j]];
}
}
}

这个求单个欧拉函数的方法

通式:φ(x)=x(1-1/p1)(1-1/p2)(1-1/p3)(1-1/p4)…..(1-1/pn)

其中p1, p2……pn为x的所有质因数,x是不为0的整数。

特殊:φ(1)=1(唯一和1互质的数(小于等于1)就是1本身)

证明:(容斥原理)

p1,p2,p3...pk为n的质因子。

与n不互质的数的个数为:
n/p1+n/p2+...+n/pk-n/(p1p2)-...-n/(pk-1pk)-n/(p1p2p3)-...-n/(pk-2pk-1pk)-... +n/(p1p2...*pk)

所以与n互质的数的个数为:
φ(n)=n-[n/p1+n/p2+...+n/pk-n/(p1p2)-...-n/(pk-1pk)-n/(p1p2p3)-...-n/(pk-2pk-1pk)-... +n/(p1p2...*pk)]=n(1-1/p1)(1-1/p2)...(1-1/pk);

inline int eular(int n)
{
int ret=n;
for(int i=2;i<=sqrt(n);i++)
{
if(n%i==0)
{
ret=ret/i*(i-1);
while(n%i==0)n/=i;
}
}
if(n>1)ret=ret/n*(n-1);
return ret;
}



快速幂与慢速乘法

这两种算法都运用到了一些分治的思想,并且都为了防止溢出而运用到了模的定理进行步步求模


快速幂

inline long long Pow(long long x,long long y)
{
long long ans=1;
for(long long i=y;i;i>>=1)
{
if(i&1)ans=ans*x%mod;
x=x*x%mod;
}
return ans%mod;
}

慢速乘法

inline long long Mul(long long x,long long y)
{
long long ans=0;
for(long long i=y;i;i>>=1)
{
if(i&1) ans=(ans+x)%mod;
x=(x+x)%mod;
}
return ans%mod;
}

还有一种比较暴力的溢出式取模,我不知道稳定性怎样,但是试过一些数据都是对的,应该速度会快一些233333

inline long long Mul(long long a,long long b,long long mod)
{
a%=mod,b%=mod;
long long c=(long double)a*b/mod;
long long ret=a*b-c*mod;
if(ret<0) ret+=mod;
else if(ret>=mod) ret-=mod;
return ret;
}

也可以将快速幂与慢速乘法结合,解决更大的数据

inline long long Pow(long long x,long long y)
{
long long ans=1;
for(long long i=y;i;i>>=1)
{
if(i&1)ans=Mul(ans,x)%mod;
x=Mul(x,x)%mod;
}
return ans%mod;
}



组合数学

百度文库

百度百科


错排公式

错排公式:F[n]=(n-1)*(F[n-1]+F[n-2])

证明:

当n个编号元素放在n个编号位置,元素编号与位置编号各不对应的方法数用D(n)表示,那么D(n-1)就表示n-1个编号元素放在n-1个编号位置,各不对应的方法数,其它类推.

第一步,把第n个元素放在一个位置,比如位置k,一共有n-1种方法;

第二步,放编号为k的元素,这时有两种情况:⑴把它放到位置n,那么,对于剩下的n-1个元素,由于第k个元素放到了位置n,剩下n-2个元素就有D(n-2)种方法;⑵第k个元素不把它放到位置n,这时,对于这n-1个元素,有D(n-1)种方法;

综上得到

F(n) = (n-1)[F(n-2)+F(n-1)]

特殊地,F(1) = 0, F(2) = 1.

模板题:HDU1465

代码:

#include
#define ll long long
#define ri register int
using namespace std;
ll a,hhh[21];
int main()
{
hhh[1]=0,hhh[2]=1,hhh[3]=2;
for(ri i=4;i<=20;++i)
hhh[i]=(i-1)*(hhh[i-1]+hhh[i-2]);
while(~scanf("%lld",&a))
printf("%lld
",hhh[a]);
return 0;
}



中国剩余定理

百度百科

模板

LL prime[2333],remain[2333];
LL x,y;
inline LL exgcd(LL a,LL b,LL &x,LL &y)
{
if(!b)
{
x=1,y=0;
return a;
}
LL gcd=exgcd(b,a%b,x,y);
LL t=x;x=y,y=t-a/b*y;
return gcd;
}
inline LL crt()
{
LL m=1,ans=0;
for(ri i=1;i<=3;++i)
m*=prime[i];
for(ri i=1;i<=3;++i)
{
LL temp=m/prime[i];
exgcd(prime[i],temp,x,y);
ans=(ans+y*temp*remain[i])%m;
}
return (ans+m)%m;
}

扩展中国剩余定理

题目P4777 【模板】扩展中国剩余定理(EXCRT)

代码:

#include
#define ll long long
#define ri register int
#define rll register long long
templateinline void read(TP&x)
{
x=0;int f=1;char c=getchar();
while(c<‘0‘||c>‘9‘){if(c==‘-‘)f=-1;c=getchar();}
while(c>=‘0‘&&c<=‘9‘){x=(x<<3)+(x<<1)+(c^48);c=getchar();}
x*=f;
}
//********快读快输********
inline ll Mul(ll x,ll y,ll mod)//慢速乘法避免溢出
{
ll ans=0;
for(rll i=y;i;i>>=1)
{
if(i&1)ans=(ans+x)%mod;
x=(x+x)%mod;
}
return ans%mod;
}
ll n,x,y,ans,m;
ll prime[233333],remain[233333];
inline ll exgcd(ll a,ll b,ll &x,ll &y)//扩展欧拉定理
{
if(!b)
{
x=1,y=0;
return a;
}
ll gcd=exgcd(b,a%b,x,y);
ll t=x;x=y,y=t-a/b*y;
return gcd;
}
inline void excrt()
{
m=prime[1],ans=remain[1];
for(ri i=2;i<=n;++i)
{
ll temp=((remain[i]-ans)%prime[i]+prime[i])%prime[i];
ll gcd=exgcd(m,prime[i],x,y);
x=Mul(x,temp/gcd,prime[i]);
ans+=m*x;
m*=prime[i]/gcd;
ans=(ans+m)%m;
}
printf("%lld",ans);
}
int main()
{
read(n);
for(ri i=1;i<=n;++i)
read(prime[i]),read(remain[i]);//输入模数与余数
excrt();
return 0;
}


这里是关于数论的洛谷日报整理:

中国剩余定理

逆元

博弈论

狄利克雷卷积与莫比乌斯反演

组合





推荐阅读
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • 本文介绍了为什么要使用多进程处理TCP服务端,多进程的好处包括可靠性高和处理大量数据时速度快。然而,多进程不能共享进程空间,因此有一些变量不能共享。文章还提供了使用多进程实现TCP服务端的代码,并对代码进行了详细注释。 ... [详细]
  • Linux环境变量函数getenv、putenv、setenv和unsetenv详解
    本文详细解释了Linux中的环境变量函数getenv、putenv、setenv和unsetenv的用法和功能。通过使用这些函数,可以获取、设置和删除环境变量的值。同时给出了相应的函数原型、参数说明和返回值。通过示例代码演示了如何使用getenv函数获取环境变量的值,并打印出来。 ... [详细]
  • 本文介绍了PE文件结构中的导出表的解析方法,包括获取区段头表、遍历查找所在的区段等步骤。通过该方法可以准确地解析PE文件中的导出表信息。 ... [详细]
  • 本文介绍了一个题目的解法,通过二分答案来解决问题,但困难在于如何进行检查。文章提供了一种逃逸方式,通过移动最慢的宿管来锁门时跑到更居中的位置,从而使所有合格的寝室都居中。文章还提到可以分开判断两边的情况,并使用前缀和的方式来求出在任意时刻能够到达宿管即将锁门的寝室的人数。最后,文章提到可以改成O(n)的直接枚举来解决问题。 ... [详细]
  • 本文介绍了Python爬虫技术基础篇面向对象高级编程(中)中的多重继承概念。通过继承,子类可以扩展父类的功能。文章以动物类层次的设计为例,讨论了按照不同分类方式设计类层次的复杂性和多重继承的优势。最后给出了哺乳动物和鸟类的设计示例,以及能跑、能飞、宠物类和非宠物类的增加对类数量的影响。 ... [详细]
  • 生成式对抗网络模型综述摘要生成式对抗网络模型(GAN)是基于深度学习的一种强大的生成模型,可以应用于计算机视觉、自然语言处理、半监督学习等重要领域。生成式对抗网络 ... [详细]
  • VScode格式化文档换行或不换行的设置方法
    本文介绍了在VScode中设置格式化文档换行或不换行的方法,包括使用插件和修改settings.json文件的内容。详细步骤为:找到settings.json文件,将其中的代码替换为指定的代码。 ... [详细]
  • Nginx使用(server参数配置)
    本文介绍了Nginx的使用,重点讲解了server参数配置,包括端口号、主机名、根目录等内容。同时,还介绍了Nginx的反向代理功能。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • CSS3选择器的使用方法详解,提高Web开发效率和精准度
    本文详细介绍了CSS3新增的选择器方法,包括属性选择器的使用。通过CSS3选择器,可以提高Web开发的效率和精准度,使得查找元素更加方便和快捷。同时,本文还对属性选择器的各种用法进行了详细解释,并给出了相应的代码示例。通过学习本文,读者可以更好地掌握CSS3选择器的使用方法,提升自己的Web开发能力。 ... [详细]
  • 动态规划算法的基本步骤及最长递增子序列问题详解
    本文详细介绍了动态规划算法的基本步骤,包括划分阶段、选择状态、决策和状态转移方程,并以最长递增子序列问题为例进行了详细解析。动态规划算法的有效性依赖于问题本身所具有的最优子结构性质和子问题重叠性质。通过将子问题的解保存在一个表中,在以后尽可能多地利用这些子问题的解,从而提高算法的效率。 ... [详细]
  • 本文介绍了指针的概念以及在函数调用时使用指针作为参数的情况。指针存放的是变量的地址,通过指针可以修改指针所指的变量的值。然而,如果想要修改指针的指向,就需要使用指针的引用。文章还通过一个简单的示例代码解释了指针的引用的使用方法,并思考了在修改指针的指向后,取指针的输出结果。 ... [详细]
  • 闭包一直是Java社区中争论不断的话题,很多语言都支持闭包这个语言特性,闭包定义了一个依赖于外部环境的自由变量的函数,这个函数能够访问外部环境的变量。本文以JavaScript的一个闭包为例,介绍了闭包的定义和特性。 ... [详细]
  • SpringMVC接收请求参数的方式总结
    本文总结了在SpringMVC开发中处理控制器参数的各种方式,包括处理使用@RequestParam注解的参数、MultipartFile类型参数和Simple类型参数的RequestParamMethodArgumentResolver,处理@RequestBody注解的参数的RequestResponseBodyMethodProcessor,以及PathVariableMapMethodArgumentResol等子类。 ... [详细]
author-avatar
李-诗-妍_519
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有