【BZOJ 1951】 [Sdoi2010]古代猪文
Description
“在那山的那边海的那边有一群小肥猪。他们活泼又聪明,他们调皮又灵敏。他们自由自在生活在那绿色的大草坪,他们善良勇敢相互都关心……” ——选自猪王国民歌 很久很久以前,在山的那边海的那边的某片风水宝地曾经存在过一个猪王国。猪王国地理位置偏僻,实施的是适应当时社会的自给自足的庄园经济,很少与外界联系,商贸活动就更少了。因此也很少有其他动物知道这样一个王国。 猪王国虽然不大,但是土地肥沃,屋舍俨然。如果一定要拿什么与之相比的话,那就只能是东晋陶渊明笔下的大家想象中的桃花源了。猪王勤政爱民,猪民安居乐业,邻里和睦相处,国家秩序井然,经济欣欣向荣,社会和谐稳定。和谐的社会带给猪民们对工作火红的热情和对未来的粉色的憧憬。 小猪iPig是猪王国的一个很普通的公民。小猪今年10岁了,在大肥猪学校上小学三年级。和大多数猪一样,他不是很聪明,因此经常遇到很多或者稀奇古怪或者旁人看来轻而易举的事情令他大伤脑筋。小猪后来参加了全猪信息学奥林匹克竞赛(Pig Olympiad in Informatics, POI),取得了不错的名次,最终保送进入了猪王国大学(Pig Kingdom University, PKU)深造。 现在的小猪已经能用计算机解决简单的问题了,比如能用P++语言编写程序计算出A + B的值。这个“成就”已经成为了他津津乐道的话题。当然,不明真相的同学们也开始对他刮目相看啦~ 小猪的故事就将从此展开,伴随大家两天时间,希望大家能够喜欢小猪。 题目描述 猪王国的文明源远流长,博大精深。 iPig在大肥猪学校图书馆中查阅资料,得知远古时期猪文文字总个数为N。当然,一种语言如果字数很多,字典也相应会很大。当时的猪王国国王考虑到如果修一本字典,规模有可能远远超过康熙字典,花费的猪力、物力将难以估量。故考虑再三没有进行这一项劳猪伤财之举。当然,猪王国的文字后来随着历史变迁逐渐进行了简化,去掉了一些不常用的字。 iPig打算研究古时某个朝代的猪文文字。根据相关文献记载,那个朝代流传的猪文文字恰好为远古时期的k分之一,其中k是N的一个正约数(可以是1和N)。不过具体是哪k分之一,以及k是多少,由于历史过于久远,已经无从考证了。 iPig觉得只要符合文献,每一种能整除N的k都是有可能的。他打算考虑到所有可能的k。显然当k等于某个定值时,该朝的猪文文字个数为N / k。然而从N个文字中保留下N / k个的情况也是相当多的。iPig预计,如果所有可能的k的所有情况数加起来为P的话,那么他研究古代文字的代价将会是G的P次方。 现在他想知道猪王国研究古代文字的代价是多少。由于iPig觉得这个数字可能是天文数字,所以你只需要告诉他答案除以999911659的余数就可以了。
Input
有且仅有一行:两个数N、G,用一个空格分开。
Output
有且仅有一行:一个数,表示答案除以999911659的余数。
Sample Input
4 2
Sample Output
2048
HINT
10%的数据中,1 <= N <= 50;
20%的数据中,1 <= N <= 1000;
40%的数据中,1 <= N <= 100000;
100%的数据中,1 <= G <= 1000000000,1 <= N <= 1000000000。
题目就是如上
根据欧拉定理我们可以得到
原式为(G^(Σi|n C(n,i) mod 999911658)) mod 999911659
(a^b=a^φ(b) mod b 此处b为素数,so φ(b)=b-1;)
也就是说求解出上试在加一个快速幂就好了
如何求解上式呢?
999911658=2*3*4679*35617 是一个Square Free Number
对于这种每个素因子的次数都是1的数来说,我们可以对它分别求解,然后用中国剩余定理去合并
所以我们就利用lucas定理(lucas(n,i,p)=C(n%p,i%p)*lucas(n/p,i/p,p))去求解C对于每一个素数的解
最后再合并就好了
——感觉好NB的样子
1 #include<cstdio> 2 #define ll long long 3 const int P=999911659; 4 int t[4]={2,3,4679,35617}; 5 int fac[5][40000],M[5],G,N; 6 int pow(ll a,ll b,ll x){ 7 ll ans=1; 8 while (b>0){ 9 if (b&1==1) ans=(ans*a)%x; 10 b=b>>1;a=(a*a)%x; 11 } 12 return ans; 13 } 14 15 int C(int a,int b,int x){ 16 if (b>a) return 0; 17 return fac[x][a]*pow(fac[x][a-b]*fac[x][b],t[x]-2,t[x])%t[x]; 18 } 19 20 int lucas(int a,int b,int x){ 21 if(!b) return 1; 22 return C(a%t[x],b%t[x],x)*lucas(a/t[x],b/t[x],x)%t[x]; 23 } 24 25 void exgcd(int a,int b,ll &x,ll &y){ 26 if (b==0){x=1,y=0;return;} 27 exgcd(b,a%b,x,y); 28 int t=x;x=y,y=t-(a/b)*y; 29 } 30 31 ll CRT() 32 { 33 ll N=1; 34 ll i,d,x0,y0,ans=0; 35 for(i=0;i<4;i++) 36 N*=t[i]; 37 for(i=0;i<4;i++) 38 { 39 d=N/t[i]; 40 exgcd(d,t[i],x0,y0); 41 ans=(ans+d*x0*M[i])%N; 42 } 43 while(ans<= 0) 44 ans+=N; 45 return ans; 46 } 47 int main(){ 48 scanf("%d%d",&N,&G); 49 if (G==P) { 50 printf("0"); 51 return 0; 52 } 53 G%=P; 54 for (int i=0;i<4;i++){//这里求阶乘 55 fac[i][0]=1; 56 for (int j=1;j<=t[i];j++) fac[i][j]=(fac[i][j-1]*j)%t[i]; 57 } 58 for (int i=1;i*i<=N;i++){ 59 if(N%i==0){ 60 int tmp=N/i; 61 for (int j=0;j<4;j++){ 62 if (tmp!=i) M[j]=(M[j]+lucas(N,i,j))%t[j]; 63 M[j]=(M[j]+lucas(N,tmp,j))%t[j]; 64 } 65 } 66 } 67 printf("%d",pow(G,CRT(),P)); 68 }