数论模板
1.exgcd
#define ll long long void exgcd(ll a,ll b) { if(b==0) { x=1,y=0;gcd=a; return; }exgcd(b,a%b); ll tmp=x,x=y,y=tmp-a/b*y; }
2.素数
#define N 1000007 void prime() { for(int i=2;i<=N;i++) { if(!no[i]) prime[++tot]=i; for(int j=1;j<=tot;j++) { if(i*prime[j]>=N)break; no[i*prime[j]]=1; if(i%prime[j]==0) break; } } }
#include<iostream> #include<cstdio> #include<cstring> #include<ctime> #include<cstdlib> #define ll long long #define N 100010 using namespace std; ll p,a; ll kfc(ll a,ll b) { ll res=0; a%=p;b%=p; while(b) { if(b&1) b--,res+=a,res%=p; a<<=1,a%=p,b>>=1; }return res%p; } ll ksm(ll a,ll b) { ll res=1; while(b) { if(b&1) res=kfc(res,a)%p; b>>=1; a=kfc(a,a)%p; }return res; } int main() { srand(time(0)); cin>>p; for(ll i=1;i<=15;i++) { a=rand()%(p-1)+1; ll x=ksm(a,p-1); if(x!=1) { printf("NO\n"); return 0; } } printf("Yes\n"); return 0; }
3.质因数分解
3. #include<cstdio> #include<cstring> #include<algorithm> using namespace std; int n; int a[10000]; int main() { scanf("%d",&n); printf("%d=",n); int cnt=0; for (int i=2;i<=n;i++) { while (n%i==0&&n) { a[cnt++]=i; n/=i; } } for (int i=0;i<cnt-1;i++) { printf("%d*",a[i]); } printf("%d\n",a[cnt-1]); return 0; } 分解单个数
#include<iostream> #include<cstdio> #include<cstring> #define N 100010 using namespace std; int no[N],prime[N],cnt[N]; int tot,n; void is_prime() { for(int i=2;i<=N;i++) { if(!no[i]) prime[++tot]=i; for(int j=1;j<=tot;j++) { if(i*prime[j]>=N) break; no[i*prime[j]]=1; if(i%prime[j]==0) break; } } } int main() { is_prime();int x=1; scanf("%d",&n);tot=0; for(int i=1;i<=n;i++) { x=1; tot++;int P=prime[i]; while(x) { x=n/P; cnt[tot]+=x; P*=prime[i]; } } for(int i=1;i<=tot;i++) if(cnt[i]!=0)printf("%d ",cnt[i]); return 0; }
4.卡特兰数
1.f(n)=f(n-1)*(4*n-2)/(n+1)
2.h(0)=0,h(1)=1
h(n)= h(0)*h(n-1)+h(1)*h(n-2) + ... + h(n-1)*h(0) (n>=2)
#include <iostream> #include <stdio.h> #include <cmath> using namespace std; int a[105][105]; // int b[105]; void catalan() { int i,j,len,carry,temp; a[1][0]=b[1]=1; len=1; for(i=2;i<=100;i++) { for(j=0;j<len;j++) a[i][j]=a[i-1][j]*(4*(i-1)+2); carry=0; for(j=0;j<len;j++) { temp=a[i][j]+carry; a[i][j]=temp%10; carry=temp/10; } while(carry) { a[i][len++]=carry%10; carry/=10; } carry=0; for(j=len-1;j>=0;j--) { temp=carry*10+a[i][j]; a[i][j]=temp/(i+1); carry=temp%(i+1); } while(!a[i][len-1]) len--; b[i]=len; } } int main() { int i,n; catalan(); scanf("%d",&n); for(i=b[n]-1;i>=0;i--) printf("%d",a[n][i]); return 0; }
5.逆元
/* P3811 */ #include<bits/stdc++.h> #define N 3000007 #define ll long long using namespace std; ll n,m,p; ll inv[N]; inline ll read() { ll x=0,f=1;char c=getchar(); while(c>'9'||c<'0'){if(c=='-')f=-1;c=getchar();} while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();} return x*f; } int main() { n=read();p=read();inv[1]=1; for(int i=2;i<=n;i++) { inv[i]=(p-p/i)*inv[p%i]%p; } for(int i=1;i<=n;i++) printf("%lld\n",inv[i]); return 0; }
6.欧拉函数
int get_phi(int x) { int ret=1; for(int i=1;prime[i]*prime[i]<=x;i++) { if(x%prime[i]==0) { ret*=prime[i]-1;x/=prime[i]; while(x%prime[i]==0) x/=prime[i],ret*=prime[i]; } } if(x>1) ret*=x-1; return ret; }
void get_phi() { cnt = 0; memset(is_prime, true, sizeof(is_prime)); phi[1] = 1; for(int i=2; i<N; i++) { if(is_prime[i])///素数 { p[cnt++] = i; phi[i] = i-1;///素数的欧拉函数值是素数- 1 } for(int j=0; j<cnt; j++) { if(i*p[j] > N) break; is_prime[i*p[j]] = false;///素数的倍数,所以i*p[j]不是素数 if(i%p[j] == 0)///性质:i mod p == 0, 那么 phi(i * p) == p * phi(i) { phi[i*p[j]] = p[j] * phi[i]; break; } else phi[i*p[j]] = (p[j]-1) * phi[i];///i mod p != 0, 那么 phi(i * p) == phi(i) * (p-1) } } }
7.高斯消元
#include<iostream> #include<cstdio> #include<cstring> using namespace std; int n; double f[101][101],ans[101]; const double eps=1e-12; int dcmp(double x) { if(x<=eps && x>=-eps) return 0; return (x>0)?1:-1; } bool gauss() { for(int i=1;i<=n;i++) { int num=i; for(int j=i+1;j<=n;j++) if(dcmp(f[i][j]-f[num][i])>0) num=j; if(num!=i) for(int j=1;j<=n+1;j++) swap(f[i][j],f[num][j]); for(int j=i+1;j<=n;j++) if(dcmp(f[j][i])) { double t=f[j][i]/f[i][i]; for(int k=1;k<=n+1;k++) f[j][k]-=t*f[i][k]; } } for(int i=n;i>=1;i--) { if(dcmp(f[i][i])==0) return 0; for(int j=i+1;j<=n;j++) f[i][n+1]-=f[i][j]*ans[j]; ans[i]=f[i][n+1]/f[i][i]; }return 1; } int main() { scanf("%d",&n); for(int i=1;i<=n;i++) for(int j=1;j<=n+1;j++) scanf("%lf",&f[i][j]); if(gauss()) for(int i=1;i<=n;i++) printf("%.2lf\n",ans[i]); else puts("No Solution\n"); return 0; }
8.中国剩余定理
void exgcd(int a,int b) { if(b==0) { x=1;y=0; return; } exgcd(b,a%b); int tmp=x;x=y;y=tmp-(a/b)*y; } int CRT(int a[],int m[],int n) { int M=1,ans=0; for(int i=1;i<=n;i++) M*=m[i]; for(int i=1;i<=n;i++) { int Mi=M/m[i]; exgcd(Mi,m[i]); ans=(ans+Mi*x*a[i])%M; } if(ans<0) ans+=M; return ans; }
9.组合数
inline void combination() { for(int i=0;i<=n;i++) for(int j=0;j<=i;j++) if(!j || i==j) c[i][j]=1; else c[i][j]=c[i-1][j-1]+c[i-1][j]; c[0][0]=0; return; }
10.Lucas定理
int ksm(int a,int b,int p) { int res=1; while(b) { if(b&1) res=res*a%p; b>>=1;a=a*a%p; }return res; } int C(int n,int m,int p) { if(m>n)return 0; return fac[n]*ksm(fac[m]*fac[n-m],p-2,p)%p; } int Lus(int n,int m,int p) { if(m==0) return 1; return (C(n%p,m%p,p)*Lus(n/p,m/p,p))%p; }
折花枝,恨花枝,准拟花开人共卮,开时人去时。
怕相思,已相思,轮到相思没处辞,眉间露一丝。