大富翁国因为通货膨胀,以及假钞泛滥,政府决定推出一项新的政策:现有钞票编号范围为1到N的阶乘,但是,政府只发行编号与M!互质的钞票。房地产第一大户沙拉公主决定预测一下大富翁国现在所有真钞票的数量。现在,请你帮助沙拉公主解决这个问题,由于可能张数非常大,你只需计算出对R取模后的答案即可。R是一个质数。
标签:
线性求逆元后好神的数论啊。。。。
#include<cstdio> #include<cstring> #include<cctype> #include<algorithm> using namespace std; #define rep(i,s,t) for(register int i=s;i<=t;i++) #define dwn(i,s,t) for(register int i=s;i>=t;i--) #define clr(x,c) memset(x,c,sizeof(x)) #define ll long long int read(){ int x=0;char c=getchar(); while(!isdigit(c)) c=getchar(); while(isdigit(c)) x=x*10+c-‘0‘,c=getchar(); return x; } const int nmax=1e7+5; int prime[nmax>>3],ans[nmax],fac[nmax],inv[nmax];bool vis[nmax]; int n,m,mod,T; void init(){ int cnt=0,tmp; rep(i,2,nmax-1){ if(!vis[i]) prime[++cnt]=i; rep(j,1,cnt){ tmp=prime[j]; if((ll)i*tmp>=nmax) break; vis[i*tmp]=1; if(i%tmp==0) break; } } fac[1]=1; rep(i,2,nmax-1) fac[i]=(ll)fac[i-1]*i%mod; inv[1]=1; rep(i,2,nmax-1) inv[i]=(ll)(mod-mod/i)*inv[mod%i]%mod; ans[1]=1; rep(i,2,nmax-1){ if(!vis[i]) ans[i]=(ll)ans[i-1]*(i-1)%mod*inv[i%mod]%mod; else ans[i]=ans[i-1]; } } int main(){ T=read(),mod=read(); init(); while(T--){ int n=read(),m=read(); printf("%d\n",(ll)fac[n]*ans[m]%mod); } return 0; }
大富翁国因为通货膨胀,以及假钞泛滥,政府决定推出一项新的政策:现有钞票编号范围为1到N的阶乘,但是,政府只发行编号与M!互质的钞票。房地产第一大户沙拉公主决定预测一下大富翁国现在所有真钞票的数量。现在,请你帮助沙拉公主解决这个问题,由于可能张数非常大,你只需计算出对R取模后的答案即可。R是一个质数。
第一行为两个整数T,R。R<=10^9+10,T<=10000,表示该组中测试数据数目,R为模后面T行,每行一对整数N,M,见题目描述 m<=n
共T行,对于每一对N,M,输出1至N!中与M!素质的数的数量对R取模后的值
标签:
原文地址:http://www.cnblogs.com/fighting-to-the-end/p/5858128.html